Come faccio a convertire un NSString da CamelCase a Titlecase, 'playername' in 'giocatore Nome'?
-
15-10-2019 - |
Domanda
sto cercando il modo più semplice per convertire una stringa dal formato dorso di cammello in formato titolo caso.
Come faccio a cambiare 'playername' in 'Nome completo'?
Soluzione
NSString *str = @"playerName";
NSMutableString *str2 = [NSMutableString string];
for (NSInteger i=0; i<str.length; i++){
NSString *ch = [str substringWithRange:NSMakeRange(i, 1)];
if ([ch rangeOfCharacterFromSet:[NSCharacterSet uppercaseLetterCharacterSet]].location != NSNotFound) {
[str2 appendString:@" "];
}
[str2 appendString:ch];
}
NSLog(@"%@", str2.capitalizedString);
Altri suggerimenti
Ecco una versione più semplice Swift. Ho buttato in un prolungamento
extension String {
func stringFromCamelCase() -> String {
var string = self
string = string.stringByReplacingOccurrencesOfString("([a-z])([A-Z])", withString: "$1 $2", options: NSStringCompareOptions.RegularExpressionSearch, range: Range<String.Index>(start: string.startIndex, end: string.endIndex))
string.replaceRange(startIndex...startIndex, with: String(self[startIndex]).capitalizedString)
return string
}
}
Utilizzo:
var str = "helloWorld"
str = str.stringFromCamelCase()
Provare a usare una regex sostituire
NSString *modified = [input stringByReplacingOccurrencesOfString:@"([a-z])([A-Z])"
withString:@"$1 $2"
options:NSRegularExpressionSearch
range:NSMakeRange(0, input.length)];
Un po 'più breve, usando NSCharacterSet:
__block NSString *str = @"myVerySpecialPlayerName" ;
// split at uppercase letters
NSArray *splitString = [str componentsSeparatedByCharactersInSet:
[NSCharacterSet uppercaseLetterCharacterSet]] ;
// get the uppercase letters
NSArray *upperCaseLetters = [str componentsSeparatedByCharactersInSet:
[[NSCharacterSet uppercaseLetterCharacterSet] invertedSet]] ;
// join with two spaces
str = [splitString componentsJoinedByString:@" "] ;
__block NSInteger offset = 0 ;
// replace each second space with the missing uppercase letter
[upperCaseLetters enumerateObjectsUsingBlock:^(NSString *character, NSUInteger idx, BOOL *stop) {
if( [character length] > 0 ) {
str = [str stringByReplacingCharactersInRange:NSMakeRange(idx+offset+1, 1) withString:character] ;
offset += 2 ;
}
}] ;
// & capitalize the first one
str = [str capitalizedString] ;
NSLog(@"%@", str) ; // "My Very Special Player Name"
penso che si può affrontare questo problema con alcune espressioni regolari. Check out questa domanda simile: iPhone dev: Sostituire caratteri maiuscoli in NSString con lo spazio e downcase
Anche se un po 'lungo, ma questa categoria per NSString dovrebbe fare il trucco. Ha superato tutte le mie prove:
- (NSString *)splitOnCapital
{
// Make a index of uppercase characters
NSRange upcaseRange = NSMakeRange('A', 26);
NSIndexSet *upcaseSet = [NSIndexSet indexSetWithIndexesInRange:upcaseRange];
// Split our camecase word
NSMutableString *result = [NSMutableString string];
NSMutableString *oneWord = [NSMutableString string];
for (int i = 0; i < self.length; i++) {
char oneChar = [self characterAtIndex:i];
if ([upcaseSet containsIndex:oneChar]) {
// Found a uppercase char, now save previous word
if (result.length == 0) {
// First word, no space in beginning
[result appendFormat:@"%@", [oneWord capitalizedString]];
}else {
[result appendFormat:@" %@", oneWord];
}
// Clear previous word for new word
oneWord = [NSMutableString string];
}
[oneWord appendFormat:@"%c", oneChar];
}
// Add last word
if (oneWord.length > 0) {
[result appendFormat:@" %@", oneWord];
}
return result;
}
Ho avuto un problema simile, le risposte qui mi ha aiutato a creare una soluzione. Ho avuto una matrice che aveva una lista di etichette ho voluto visualizzare in un UITableView, un'etichetta per riga.
Il mio problema è stato Ho analizzato queste etichette da un XML restituito da un'azione di SOAP e non avevo idea sul formato delle stringhe.
In primo luogo ho implementato risposta webstersx in un metodo. Questo è stato grande, ma alcune di queste etichette è iniziato con una lettera maiuscola e alcuni casi di cammello dove (ad esempio, alcune stringhe dove exampleLabel
e altri in cui ExampleLabel
. Quindi questo significava quelli che iniziano con un capitale avevano uno spazio inserito davanti alla stringa.
Ho superato questo tagliando spazi bianchi del beggining e alla fine della stringa utilizzando stringByTrimmingCharactersInSet
di NSString.
Il problema successivo è stato abbreviazioni utilizzate, come ad esempio "ID" o "PNR Stato", in cui viene visualizzato come "ID" e "Stato PNR", come le lettere maiuscole in cui, giustamente, di essere preso e uno spazio inserito prima.
Ho superato questo problema mediante l'attuazione di una regex simile alla risposta di emdog4 nel mio nuovo metodo.
Ecco la mia soluzione completa:
- (NSString *)formatLabel:(NSString *)label
{
NSMutableString *str2 = [NSMutableString string];
for (NSInteger i=0; i<label.length; i++){
NSString *ch = [label substringWithRange:NSMakeRange(i, 1)];
if ([ch rangeOfCharacterFromSet:[NSCharacterSet uppercaseLetterCharacterSet]].location != NSNotFound) {
[str2 appendString:@" "];
}
[str2 appendString:ch];
}
NSString * formattedString = [str2 stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]].capitalizedString;
formattedString = [formattedString stringByReplacingOccurrencesOfString:@"([A-Z]) (?![A-Z][a-z])" withString:@"$1" options:NSRegularExpressionSearch range:NSMakeRange(0, formattedString.length)];
return formattedString;
}
Poi ho semplicemente chiamare qualcosa di simile, per esempio, che restituirà la mia stringa formattata:
NSString * formattedLabel = [self formatLabel:@"PNRStatus"];
NSLog(@"Formatted Label: %@", formattedLabel);
output sarà:
2013/10/10 10: 44: 39,888 Test Project [28296: A0B] formattato Etichetta: PNR Stato
Se qualcuno ha bisogno di una versione di Swift:
func camelCaseToTitleCase(s: NSString) -> String {
var newString = ""
if s.length > 0 {
newString = s.substringToIndex(1).uppercaseString
for i in 1..<s.length {
let char = s.characterAtIndex(i)
if NSCharacterSet.uppercaseLetterCharacterSet().characterIsMember(char) {
newString += " "
}
newString += s.substringWithRange(NSRange(location: i, length: 1))
}
}
return newString
}
Cercando di essere più compatibile con Unicode
extension String {
func camelCaseToTitleCase() -> String {
return unicodeScalars.map(replaceCaptialsWithSpacePlusCapital).joined().capitalized
}
private func replaceCaptialsWithSpacePlusCapital(unichar: UnicodeScalar) -> String {
if CharacterSet.uppercaseLetters.contains(unichar) {
return " \(unichar)"
}
return "\(unichar)"
}
}
Anche se tecnicamente più breve, più inefficiente
NSString *challengeString = @"playerName";
NSMutableString *rStr = [NSMutableString stringWithString:challengeString];
while ([rStr rangeOfCharacterFromSet:[NSCharacterSet uppercaseLetterCharacterSet]].location != NSNotFound) {
[rStr replaceCharactersInRange:[rStr rangeOfCharacterFromSet:[NSCharacterSet uppercaseLetterCharacterSet]] withString:[[NSString stringWithFormat:@" %@", [rStr substringWithRange:[rStr rangeOfCharacterFromSet:[NSCharacterSet uppercaseLetterCharacterSet]]]] lowercaseString]];
}
NSLog(@"%@", rStr.capitalizedString);
Non è sicuro questo è molto più breve di websterx, ma trovo usando characterIsMember più facile da leggere e capire. Anche aggiunto un controllo di lunghezza per fissare lo spazio prima se la stringa inizia con un capitale.
NSString *str = @"PlayerNameHowAboutALongerString";
NSMutableString *str2 = [NSMutableString string];
for (NSInteger i=0; i<str.length; i++){
unichar ch = [str characterAtIndex:i];
if ( [[NSCharacterSet uppercaseLetterCharacterSet] characterIsMember:ch]) {
if (str2.length > 0 ) {
[str2 appendString:@" "];
}
}
[str2 appendString:[NSString stringWithCharacters:&ch length:1]];
}
NSLog(@"--%@--", str2.capitalizedString);
La risposta accettata non ha funzionato per me perché non capitalizza la prima lettera, e se la prima lettera è già in maiuscolo, si aggiunge uno spazio estraneo agli inizi. Ecco la mia versione migliorata:
- (NSString *)titleFromCamelCaseString:(NSString *)input
{
NSMutableString *output = [NSMutableString string];
[output appendString:[[input substringToIndex:1] uppercaseString]];
for (NSUInteger i = 1; i < [input length]; i++)
{
unichar character = [input characterAtIndex:i];
if ([[NSCharacterSet uppercaseLetterCharacterSet] characterIsMember:character])
{
[output appendString:@" "];
}
[output appendFormat:@"%C", character];
}
return output;
}
Ecco Swift Code (codice Objective C per webstersx), Grazie!
var str: NSMutableString = "iLoveSwiftCode"
var str2: NSMutableString = NSMutableString()
for var i:NSInteger = 0 ; i < str.length ; i++ {
var ch:NSString = str.substringWithRange(NSMakeRange(i, 1))
if(ch .rangeOfCharacterFromSet(NSCharacterSet.uppercaseLetterCharacterSet()).location != NSNotFound) {
str2 .appendString(" ")
}
str2 .appendString(ch)
}
println("\(str2.capitalizedString)")
}
NSString *input = @"playerName";
NSString *modified = [input stringByReplacingOccurrencesOfString:@"(?<!^)[A-Z]" withString:@" $0" options:NSRegularExpressionSearch range:NSMakeRange(0, input.length)].capitalizedString;
Un'altra soluzione sotto Swift 2.2
extension String {
var stringFromCamelCase:String {
return (self as NSString).replacingOccurrences(
of: "([a-z])([A-Z])",
with: "$1 $2",
options: CompareOptions.regularExpressionSearch,
range: NSMakeRange(0, self.characters.count)
).uppercaseFirst
}
var uppercaseFirst: String {
return String(characters.prefix(1)).uppercased() + String(characters.dropFirst()).lowercased()
}
}
provare a utilizzare:
string.Split()
quindi utilizzare la lettera tappo in segno