Objective-Cに(stringByAppendingString:
)文字列連結へのショートカット、またはNSString
を扱うためのショートカットはありますか?
例えば、私は作りたいのですが:
NSString *myString = @"This";
NSString *test = [myString stringByAppendingString:@" is just a test"];
もっと似たようなもの:
string myString = "This";
string test = myString + " is just a test";
私が考えることができる2つの答え...どちらも特に連結演算子を持つことほど特に楽しいものではありません。
まず、NSMutableString
メソッドを使用するappendString
を使用します。余分な一時文字列が不要になります。
次に、NSArray
メソッドを使って連結するためにcomponentsJoinedByString
を使います。
[NSString stringWithFormat:@"%@/%@/%@", one, two, three];
複数の追加(a + b + c + d)に満足していないと思います。
NSLog(@"%@", [Util append:one, @" ", two, nil]); // "one two"
NSLog(@"%@", [Util append:three, @"/", two, @"/", one, nil]); // three/two/one
のようなものを使う
+ (NSString *) append:(id) first, ...
{
NSString * result = @"";
id eachArg;
va_list alist;
if(first)
{
result = [result stringByAppendingString:first];
va_start(alist, first);
while (eachArg = va_arg(alist, id))
result = [result stringByAppendingString:eachArg];
va_end(alist);
}
return result;
}
2つのNSString リテラル がある場合は、これを実行することもできます。
NSString *joinedFromLiterals = @"ONE " @"MILLION " @"YEARS " @"DUNGEON!!!";
#definesに参加するのにも便利です。
#define STRINGA @"Also, I don't know "
#define STRINGB @"where food comes from."
#define JOINED STRINGA STRINGB
楽しい。
私はこの記事に戻り続け、必要に応じてできるだけ多くの変数で機能するこの簡単な解決策を見つけるために答えをソートしていきます。
[NSString stringWithFormat:@"%@/%@/%@", three, two, one];
例えば:
NSString *urlForHttpGet = [NSString stringWithFormat:@"http://example.com/login/username/%@/userid/%i", userName, userId];
コロンは特殊記号の一種ですが、 は メソッドシグネチャの一部であるため、_ NSString
カテゴリを削除してnon-idiomaticスタイルの文字列連結を追加することができます。
[@"This " : @"feels " : @"almost like " : @"concatenation with operators"];
あなたが役に立つと思うだけの数のコロンで区切られた引数を定義することができます... ;-)
念のため、nil
で終わる文字列のリストを取る可変引数を持つconcat:
も追加しました。
// NSString+Concatenation.h
#import <Foundation/Foundation.h>
@interface NSString (Concatenation)
- (NSString *):(NSString *)a;
- (NSString *):(NSString *)a :(NSString *)b;
- (NSString *):(NSString *)a :(NSString *)b :(NSString *)c;
- (NSString *):(NSString *)a :(NSString *)b :(NSString *)c :(NSString *)d;
- (NSString *)concat:(NSString *)strings, ...;
@end
// NSString+Concatenation.m
#import "NSString+Concatenation.h"
@implementation NSString (Concatenation)
- (NSString *):(NSString *)a { return [self stringByAppendingString:a];}
- (NSString *):(NSString *)a :(NSString *)b { return [[self:a]:b];}
- (NSString *):(NSString *)a :(NSString *)b :(NSString *)c
{ return [[[self:a]:b]:c]; }
- (NSString *):(NSString *)a :(NSString *)b :(NSString *)c :(NSString *)d
{ return [[[[self:a]:b]:c]:d];}
- (NSString *)concat:(NSString *)strings, ...
{
va_list args;
va_start(args, strings);
NSString *s;
NSString *con = [self stringByAppendingString:strings];
while((s = va_arg(args, NSString *)))
con = [con stringByAppendingString:s];
va_end(args);
return con;
}
@end
// NSString+ConcatenationTest.h
#import <SenTestingKit/SenTestingKit.h>
#import "NSString+Concatenation.h"
@interface NSString_ConcatenationTest : SenTestCase
@end
// NSString+ConcatenationTest.m
#import "NSString+ConcatenationTest.h"
@implementation NSString_ConcatenationTest
- (void)testSimpleConcatenation
{
STAssertEqualObjects([@"a":@"b"], @"ab", nil);
STAssertEqualObjects([@"a":@"b":@"c"], @"abc", nil);
STAssertEqualObjects([@"a":@"b":@"c":@"d"], @"abcd", nil);
STAssertEqualObjects([@"a":@"b":@"c":@"d":@"e"], @"abcde", nil);
STAssertEqualObjects([@"this " : @"is " : @"string " : @"concatenation"],
@"this is string concatenation", nil);
}
- (void)testVarArgConcatenation
{
NSString *concatenation = [@"a" concat:@"b", nil];
STAssertEqualObjects(concatenation, @"ab", nil);
concatenation = [concatenation concat:@"c", @"d", concatenation, nil];
STAssertEqualObjects(concatenation, @"abcdab", nil);
}
メソッドを作成します。
- (NSString *)strCat: (NSString *)one: (NSString *)two
{
NSString *myString;
myString = [NSString stringWithFormat:@"%@%@", one , two];
return myString;
}
それから、あなたがそれを必要とするどんな関数でも、あなたの文字列またはテキストフィールドを設定するか、あるいは何でもこの関数の戻り値に設定してください。
または、ショートカットを作成するには、NSStringをC++の文字列に変換して、そこで+を使用します。
このようにしてください。
NSString *string1, *string2, *result;
string1 = @"This is ";
string2 = @"my string.";
result = [result stringByAppendingString:string1];
result = [result stringByAppendingString:string2];
OR
result = [result stringByAppendingString:@"This is "];
result = [result stringByAppendingString:@"my string."];
マクロ:
// stringConcat(...)
// A shortcut for concatenating strings (or objects' string representations).
// Input: Any number of non-nil NSObjects.
// Output: All arguments concatenated together into a single NSString.
#define stringConcat(...) \
[@[__VA_ARGS__] componentsJoinedByString:@""]
テストケース:
- (void)testStringConcat {
NSString *actual;
actual = stringConcat(); //might not make sense, but it's still a valid expression.
STAssertEqualObjects(@"", actual, @"stringConcat");
actual = stringConcat(@"A");
STAssertEqualObjects(@"A", actual, @"stringConcat");
actual = stringConcat(@"A", @"B");
STAssertEqualObjects(@"AB", actual, @"stringConcat");
actual = stringConcat(@"A", @"B", @"C");
STAssertEqualObjects(@"ABC", actual, @"stringConcat");
// works on all NSObjects (not just strings):
actual = stringConcat(@1, @" ", @2, @" ", @3);
STAssertEqualObjects(@"1 2 3", actual, @"stringConcat");
}
代替マクロ: (最小数の引数を強制したい場合)
// stringConcat(...)
// A shortcut for concatenating strings (or objects' string representations).
// Input: Two or more non-nil NSObjects.
// Output: All arguments concatenated together into a single NSString.
#define stringConcat(str1, str2, ...) \
[@[ str1, str2, ##__VA_ARGS__] componentsJoinedByString:@""];
Webサービスへのリクエストを作成するとき、私は次のようなことをするのがとても簡単であり、Xcodeで連結を読みやすくすると思います。
NSString* postBody = {
@"<?xml version=\"1.0\" encoding=\"utf-8\"?>"
@"<soap:Envelope xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" xmlns:soap=\"http://schemas.xmlsoap.org/soap/envelope/\">"
@" <soap:Body>"
@" <WebServiceMethod xmlns=\"\">"
@" <parameter>test</parameter>"
@" </WebServiceMethod>"
@" </soap:Body>"
@"</soap:Envelope>"
};
AppendString(AS)マクロを作成することによるショートカット...
#AS(A、B)を定義します。[(A)stringByAppendingString:(B)]
NSString * myString = @ "This"; NSString * test = AS(myString、@ "は単なるテストです);
注意:
マクロを使うのであれば、もちろん可変引数でそれをやるだけで、EthanBの答えを見てください。
NSString *label1 = @"Process Name: ";
NSString *label2 = @"Process Id: ";
NSString *processName = [[NSProcessInfo processInfo] processName];
NSString *processID = [NSString stringWithFormat:@"%d", [[NSProcessInfo processInfo] processIdentifier]];
NSString *testConcat = [NSString stringWithFormat:@"%@ %@ %@ %@", label1, processName, label2, processID];
新しい配列リテラル構文を使った簡単な方法は次のとおりです。
NSString * s = [@[@"one ", @"two ", @"three"] componentsJoinedByString:@""];
^^^^^^^ create array ^^^^^
^^^^^^^ concatenate ^^^^^
NSString *myString = @"This";
NSString *test = [myString stringByAppendingString:@" is just a test"];
Objective Cを使用して数年経った今、Objective Cを使用して達成しようとしていることを達成するためのこれが最善の方法だと思います。
Xcodeアプリケーションで「N」の入力を開始すると、「NSString」に自動補完されます。 「str」と入力すると、「stringByAppendingString」に自動補完されます。そのため、キーストロークはかなり限られています。
"@"キーを押して、読みやすいコードを書くプロセスをタブ操作する手間が省けたら、もう問題にはなりません。それは適応の問題です。
NSString *result=[NSString stringWithFormat:@"%@ %@", @"Hello", @"World"];
stringByAppendingString
を短くして #define を使うのはどうでしょうか。
#define and stringByAppendingString
したがって、あなたは使うだろう:
NSString* myString = [@"Hello " and @"world"];
問題はそれが2つの文字列に対してのみ機能するということです、あなたはより多くの追加のために追加の括弧をラップすることを要求されます:
NSString* myString = [[@"Hello" and: @" world"] and: @" again"];
c = [a stringByAppendingString: b]
をもっと短くする唯一の方法は、st
の点でオートコンプリートを使うことです。 +
演算子はCの一部であり、Objective-Cオブジェクトについては知りません。
私はこのコードを試しました。それは私のために働いた。
NSMutableString * myString=[[NSMutableString alloc]init];
myString=[myString stringByAppendingString:@"first value"];
myString=[myString stringByAppendingString:@"second string"];
lldb
ペインで以下を試していました
[NSString stringWithFormat:@"%@/%@/%@", three, two, one];
どのエラー.
代わりにallocとinitWithFormat
メソッドを使用してください。
[[NSString alloc] initWithFormat:@"%@/%@/%@", @"three", @"two", @"one"];
これはより良いロギングのためのもので、ロギングのみ - diciusの優れた複数引数方式に基づいています。 Loggerクラスを定義し、それを次のように呼び出します。
[Logger log: @"foobar ", @" asdads ", theString, nil];
Var argsを "nil"で終わらなければならないことを除けば、ほとんど問題ありませんが、Objective-Cではそれを回避する方法はないと思います。
Logger.h
@interface Logger : NSObject {
}
+ (void) log: (id) first, ...;
@end
Logger.m
@implementation Logger
+ (void) log: (id) first, ...
{
// TODO: make efficient; handle arguments other than strings
// thanks to @diciu http://stackoverflow.com/questions/510269/how-do-i-concatenate-strings-in-objective-c
NSString * result = @"";
id eachArg;
va_list alist;
if(first)
{
result = [result stringByAppendingString:first];
va_start(alist, first);
while (eachArg = va_arg(alist, id))
{
result = [result stringByAppendingString:eachArg];
}
va_end(alist);
}
NSLog(@"%@", result);
}
@end
concat stringsのみを扱うために、NSStringにCategoryを定義し、それに文字列を返すこと以外は上記のlogメソッドとまったく同じように見える静的(+)連結メソッドを追加します。これは文字列メソッドなのでNSStringにあり、1-N文字列から新しい文字列を作成したいという理由で静的であり、appendの一部である文字列のいずれかに対してそれを呼び出すことはしません。
NSNumber *lat = [NSNumber numberWithDouble:destinationMapView.camera.target.latitude];
NSNumber *lon = [NSNumber numberWithDouble:destinationMapView.camera.target.longitude];
NSString *DesconCatenated = [NSString stringWithFormat:@"%@|%@",lat,lon];
私の好ましい方法はこれです:
NSString *firstString = @"foo";
NSString *secondString = @"bar";
NSString *thirdString = @"baz";
NSString *joinedString = [@[firstString, secondString, thirdString] join];
これを実現するには、NSArrayにjoinメソッドをカテゴリとともに追加します。
#import "NSArray+Join.h"
@implementation NSArray (Join)
-(NSString *)join
{
return [self componentsJoinedByString:@""];
}
@end
@[]
はNSArray
の短い定義です。これが文字列を連結するための最速の方法だと思います。
カテゴリを使用したくない場合は、直接componentsJoinedByString:
メソッドを使用してください。
NSString *joinedString = [@[firstString, secondString, thirdString] componentsJoinedByString:@""];
StringWithFormatを試してください。
NSString *myString = [NSString stringWithFormat:@"%@ %@ %@ %d", "The", "Answer", "Is", 42];
文字列を扱うときに、ソースファイルをObjC++にする方が簡単な場合が多いので、質問に示されている2番目の方法を使用してstd :: stringsを連結できます。
std::string stdstr = [nsstr UTF8String];
//easier to read and more portable string manipulation goes here...
NSString* nsstr = [NSString stringWithUTF8String:stdstr.c_str()];
あなたはNSArrayを使うことができます
NSString *string1=@"This"
NSString *string2=@"is just"
NSString *string3=@"a test"
NSArray *myStrings = [[NSArray alloc] initWithObjects:string1, string2, string3,nil];
NSString *fullLengthString = [myStrings componentsJoinedByString:@" "];
または
あなたが使用することができます
NSString *imageFullName=[NSString stringWithFormat:@"%@ %@ %@.", string1,string2,string3];
私がテストしたとき、これらのフォーマットのどちらもXCode7で動きます:
NSString *sTest1 = {@"This" " and that" " and one more"};
NSString *sTest2 = {
@"This"
" and that"
" and one more"
};
NSLog(@"\n%@\n\n%@",sTest1,sTest2);
何らかの理由で、ミックスの最初の文字列には@演算子文字しか必要ありません。
ただし、変数挿入では機能しません。そのためには、「 - 」の代わりに「cat」でマクロを使用することを除いて、 この非常に単純な解決策 を使用できます。
UIテストでこれを必要とするすべてのObjective C愛好家のために:
-(void) clearTextField:(XCUIElement*) textField{
NSString* currentInput = (NSString*) textField.value;
NSMutableString* deleteString = [NSMutableString new];
for(int i = 0; i < currentInput.length; ++i) {
[deleteString appendString: [NSString stringWithFormat:@"%c", 8]];
}
[textField typeText:deleteString];
}
listOfCatalogIDs =[@[@"id[]=",listOfCatalogIDs] componentsJoinedByString:@""];
Uがそこにいくつの文字列があるかわからないと想像しましょう。
NSMutableArray *arrForStrings = [[NSMutableArray alloc] init];
for (int i=0; i<[allMyStrings count]; i++) {
NSString *str = [allMyStrings objectAtIndex:i];
[arrForStrings addObject:str];
}
NSString *readyString = [[arrForStrings mutableCopy] componentsJoinedByString:@", "];
In スイフト
let str1 = "This"
let str2 = "is just a test"
var appendStr1 = "\(str1) \(str2)" // appendStr1 would be "This is just a test"
var appendStr2 = str1 + str2 // // appendStr2 would be "This is just a test"
また、+=
演算子を以下と同じように使用できます。
var str3 = "Some String"
str3 += str2 // str3 would be "Some String is just a test"