1179

Are there any shortcuts to (stringByAppendingString:) string concatenation in Objective-C, or shortcuts for working with NSString in general?

For example, I'd like to make:

NSString *myString = @"This"; NSString *test = [myString stringByAppendingString:@" is just a test"]; 

something more like:

string myString = "This"; string test = myString + " is just a test"; 
19
  • 7
    I'd just like to propose '@+' as a concatenation operator. I'll expect this in the next update to Objective-C, kthxbai Commented Jan 15, 2013 at 20:15
  • 51
    @NicolasMiari This is not the only feature that Objective-C lacks. There are dozens of others. Quoting from the link Jordão posted: "Objective-C is, to be blunt, a primitive language. Compare it to any modern language and you quickly find it lacking." I agree. Objective-C (early 1980s) is C (early 1970s) with the addition of a very simple and not very type-safe kind of OOP. It's ok, but compared with Java or C#, it feels very old-fashioned. Commented Aug 15, 2013 at 9:13
  • 6
    @NicolasMiari: Interpreted language? C# and Java are compiled languages. Compiled to bytecode, which is then compiled again to machine code. Commented Sep 16, 2013 at 11:20
  • 4
    Things are changed now : Swift (apple new language) is more straightforward Commented Jun 11, 2014 at 19:45
  • 7
    Regarding "type safety", I think it's a problem of style. For someone coming from C#/C++, having heterogeneous arrays of objects of any type might seem odd, but for someone accustomed to Objective-C/Cocoa, it's a form of dynamism and freedom; it is an advantage provided you know what you are doing. Like pretty much everything about C that younger programmers bash these days... Commented Jul 18, 2014 at 1:11

30 Answers 30

1187

An option:

[NSString stringWithFormat:@"%@/%@/%@", one, two, three]; 

Another option:

I'm guessing you're not happy with multiple appends (a+b+c+d), in which case you could do:

NSLog(@"%@", [Util append:one, @" ", two, nil]); // "one two" NSLog(@"%@", [Util append:three, @"/", two, @"/", one, nil]); // three/two/one 

using something like

+ (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; } 
Sign up to request clarification or add additional context in comments.

4 Comments

@pablasso Agreed. The Util method is pretty ugly. If you wanted such a thing, it should be done as a NSString category with a name like +stringByAppendingStrings:. Even a straight-up function with a name like NSStringForAppendedStrings(...) would be better than a static method in a class like Util (anything with "Util" in the name is likely poorly factored). The function is also better implemented with an NSMutableString and -appendString to avoid creating an unbounded set of temporary autoreleased NSStrings.
With large strings this can waste memory. Recommended is something more like StringBuilder in real programming languages. Then you could just figure out how much memory is needed before you actually start appending. The above approach could be refactored to do this. However, it would be better to make a StringBuilder object as this would save the user of it from having to keep track of a list of all the strings they need combined.
how do you import Util? this IDE is frustrating (no suggested "import something.Util" like on eclipse, and I find no mention of "Util" anywhere. Is this a class I'm supposed to code myself?
stringWithFormat is not only very elegant, it is also much more powerful. You use it with @"%@%@" to concatenate two strings, @"%@%@%@" to concatenate three strings, but you can put any extra characters inside, print numbers, reorder parameters if you like and so on. The format string can be localised, making it ten times more powerful. String concatenation is for beginners.
633

Two answers I can think of... neither is particularly as pleasant as just having a concatenation operator.

First, use an NSMutableString, which has an appendString method, removing some of the need for extra temp strings.

Second, use an NSArray to concatenate via the componentsJoinedByString method.

7 Comments

Although the other option has many upvotes, I think this is the best answer if you don't know all your strings upon construction. Every time you append a string, you're creating a lot of overhead. Using a mutable string removes that problem.
+1 Agree w @Eli. These are generally the best solutions. NSArray -componentsJoinedByString can be done in a single line pretty well: string = [[NSArray arrayWithObjects:@"This", "Is", "A", "Test", nil] componentsJoinedByString:@" "];
+1 for this answer. [NSMutableString appendString] is more memory friendly than [NSString stringByAppendingStrings].
@RobNapier:Now with the new array literal syntax, it is even better.
The [NSString stringWithFormat:@"%@/%@/%@", three, two, one]; technique seems the most elegant. It should be the selected answer.
|
156

If you have 2 NSString literals, you can also just do this:

NSString *joinedFromLiterals = @"ONE " @"MILLION " @"YEARS " @"DUNGEON!!!"; 

That's also useful for joining #defines:

#define STRINGA @"Also, I don't know " #define STRINGB @"where food comes from." #define JOINED STRINGA STRINGB 

Enjoy.

7 Comments

@CristiBăluță :) But this works only with literals not with dynamically created NSString instances.
You actually don't need the @s on the strings after the first. @"I" " really" " enjoy"...
You should probably want to put STRINGA and STRINGB within parenthesis, otherwise you might get weird results when the macro is resolved. #define JOINED (STRINGA STRINGB)
@JohannesFahrenkrug Then why this NSString* const SQL_CREATE_TABLE_str = @"CREATE TABLE IF NOT EXISTS " TABLE_NAME @" (...);"; doesn't work? I'm having Expected '@' in program error :(
@Vagif how is TABLE_NAME defined?
|
86

I keep returning to this post and always end up sorting through the answers to find this simple solution that works with as many variables as needed:

[NSString stringWithFormat:@"%@/%@/%@", three, two, one]; 

For example:

NSString *urlForHttpGet = [NSString stringWithFormat:@"http://example.com/login/username/%@/userid/%i", userName, userId]; 

1 Comment

Thx for posting this! I'm completely new to Objective-C and I thought that stringWithFormat was a variable name, hahahah. Thx for the example!
57

Create a method:

- (NSString *)strCat: (NSString *)one: (NSString *)two { NSString *myString; myString = [NSString stringWithFormat:@"%@%@", one , two]; return myString; } 

Then, in whatever function you need it in, set your string or text field or whatever to the return value of this function.

Or, to make a shortcut, convert the NSString into a C++ string and use the '+' there.

Comments

47

Well, as colon is kind of special symbol, but is part of method signature, it is possible to exted the NSString with category to add this non-idiomatic style of string concatenation:

[@"This " : @"feels " : @"almost like " : @"concatenation with operators"]; 

You can define as many colon separated arguments as you find useful... ;-)

For a good measure, I've also added concat: with variable arguments that takes nil terminated list of strings.

// 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); } 

2 Comments

I downvoted this a year ago because it's not a very good answer. To cope with concatenating a large number of strings, Palimondo's implementation requires either implementing a large number of very similar looking methods, or calling the methods several times, resulting in a large chunk of code that essentially just concatenates strings. Using this approach, you don't get any benefit over a simple stringWithFormat:. Not to mention the lack of named parameters which is not only non-standard but also confusing.
The original asker mentioned stringByAppendingString, and he never said anything about using more than two arguments. I like this answer better than the accepted one. It's pretty clever.
40

Use stringByAppendingString: this way:

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."]; 

1 Comment

You do realize that you're suggesting the exact thing he wanted not to do, right?
32

Macro:

// 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:@""] 

Test Cases:

- (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"); } 

Alternate macro: (if you wanted to enforce a minimum number of arguments)

// 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:@""]; 

2 Comments

Haven't checked this question in a while, but I'm leaning towards accepting this as the right answer after all these years!
This also arguably has better behaviour than -[NSString stringByAppendingString:] for this use case — with the former you'll get an exception if the argument is nil but not if the receiver is. So it's conceivably 50% odds on a mistake in your string feeder failing silently and 50% on an exception. With stringConcat you're guaranteed an exception on any nil, anywhere in the list. Which at least is more predictable.
27

When building requests for web services, I find doing something like the following is very easy and makes concatenation readable in 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>" }; 

2 Comments

For an objective-c noob can you explain what this syntax is doing? Is this creating an array of strings and joining them somewhow? A reference to any docs would be cool too.
@NormanH: This is actually part of the C language. After a little digging, I was able to find this. It states under the "String concatenation" phase: All adjacent string and wide-string literals are concatenated. For example, "String " "concatenation" becomes "String concatenation".
27

Shortcut by creating AppendString (AS) macro ...

 #define AS(A,B) [(A) stringByAppendingString:(B)]
NSString *myString = @"This"; NSString *test = AS(myString,@" is just a test");

Note:

If using a macro, of course just do it with variadic arguments, see EthanB's answer.

5 Comments

Cool! I still think the Util above is a much more elegant solution; you can append only one string with this macro, right?
True, the AS macro above does one append per line of code. If multiple appends are a common need, then more macros can be created. For example, a macro to append two strings: <pre> #define A2S(A,B,C) [[(A) stringByAppendingString:(B)] stringByAppendingString:(C)] </pre>
Or, simply shorten the typing required with a macro like "#define AS stringByAppendingString", then just use "AS" where your would normally type "stringByAppendingString", and enjoy multiple appends per line of code.
The problem with these macros is that they undermine one of the major goals of Objective-C, which is readability. It's extremely unclear what "AS" does. Saving a few keystrokes (most of which are handled with autocompletion) at the expense of readability is seldom a good trade-off. There are exceptions (the @"" syntax is much more readable than having to use +stringWithUTF8String: every time), but the goal should still be readability rather than simply brevity. You write once, but you debug forever.
Hi Rob - I can't agree with you in this. Sure, "AS" is a bad name. It should be named perhaps "CAT".
13
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]; 

Comments

11

Here's a simple way, using the new array literal syntax:

NSString * s = [@[@"one ", @"two ", @"three"] componentsJoinedByString:@""]; ^^^^^^^ create array ^^^^^ ^^^^^^^ concatenate ^^^^^ 

Comments

9
NSString *myString = @"This"; NSString *test = [myString stringByAppendingString:@" is just a test"]; 

After a couple of years now with Objective C I think this is the best way to work with Objective C to achieve what you are trying to achieve.

Start keying in "N" in your Xcode application and it autocompletes to "NSString". key in "str" and it autocompletes to "stringByAppendingString". So the keystrokes are quite limited.

Once you get the hang of hitting the "@" key and tabbing the process of writing readable code no longer becomes a problem. It is just a matter of adapting.

1 Comment

You do realize that you're suggesting the exact thing he wanted not to do, right?
8

The only way to make c = [a stringByAppendingString: b] any shorter is to use autocomplete at around the st point. The + operator is part of C, which doesn't know about Objective-C objects.

1 Comment

You do realize that you're suggesting the exact thing he wanted not to do, right? At the very least, a #define could be used to shorten it.
8

How about shortening stringByAppendingString and use a #define:

#define and stringByAppendingString 

Thus you would use:

NSString* myString = [@"Hello " and @"world"]; 

Problem is that it only works for two strings, you're required to wrap additional brackets for more appends:

NSString* myString = [[@"Hello" and: @" world"] and: @" again"]; 

1 Comment

XCode7 won't let you use this option anymore -- says "and" is a reserved word. However, you can use "cat" instead, for conCATenation. I did and your solution works perfectly and is extremely simple.
8
NSString *result=[NSString stringWithFormat:@"%@ %@", @"Hello", @"World"]; 

Comments

7
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]; 

Comments

6

I tried this code. it's worked for me.

NSMutableString * myString=[[NSMutableString alloc]init]; myString=[myString stringByAppendingString:@"first value"]; myString=[myString stringByAppendingString:@"second string"]; 

2 Comments

This is the worst abuse of an NSMutableString ever. The whole point of NSMutableString is that you don't need methods creating autoreleased strings but can modify the string itself.
Agree. At least use appendString: if you're using a NSMutableString.
6

Was trying the following in the lldb pane

[NSString stringWithFormat:@"%@/%@/%@", three, two, one]; 

which errors.

instead use alloc and initWithFormat method:

[[NSString alloc] initWithFormat:@"%@/%@/%@", @"three", @"two", @"one"]; 

1 Comment

...wish I had enough reputation to comment but hope this helps somebody else.
4

This is for better logging, and logging only - based on dicius excellent multiple argument method. I define a Logger class, and call it like so:

[Logger log: @"foobar ", @" asdads ", theString, nil]; 

Almost good, except having to end the var args with "nil" but I suppose there's no way around that in 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 

In order to only concat strings, I'd define a Category on NSString and add a static (+) concatenate method to it that looks exactly like the log method above except it returns the string. It's on NSString because it's a string method, and it's static because you want to create a new string from 1-N strings, not call it on any one of the strings that are part of the append.

Comments

4
NSNumber *lat = [NSNumber numberWithDouble:destinationMapView.camera.target.latitude]; NSNumber *lon = [NSNumber numberWithDouble:destinationMapView.camera.target.longitude]; NSString *DesconCatenated = [NSString stringWithFormat:@"%@|%@",lat,lon]; 

Comments

3

Try stringWithFormat:

NSString *myString = [NSString stringWithFormat:@"%@ %@ %@ %d", "The", "Answer", "Is", 42]; 

1 Comment

Why does this have 2 downvotes? Is it because this was already mentioned in another answer?
3

When dealing with strings often I find it easier to make the source file ObjC++, then I can concatenate std::strings using the second method shown in the question.

std::string stdstr = [nsstr UTF8String]; //easier to read and more portable string manipulation goes here... NSString* nsstr = [NSString stringWithUTF8String:stdstr.c_str()]; 

Comments

3

My preferred method is this:

NSString *firstString = @"foo"; NSString *secondString = @"bar"; NSString *thirdString = @"baz"; NSString *joinedString = [@[firstString, secondString, thirdString] join]; 

You can achieve it by adding the join method to NSArray with a category:

#import "NSArray+Join.h" @implementation NSArray (Join) -(NSString *)join { return [self componentsJoinedByString:@""]; } @end 

@[] it's the short definition for NSArray, I think this is the fastest method to concatenate strings.

If you don't want to use the category, use directly the componentsJoinedByString: method:

NSString *joinedString = [@[firstString, secondString, thirdString] componentsJoinedByString:@""]; 

Comments

3

You can use NSArray as

NSString *string1=@"This" NSString *string2=@"is just" NSString *string3=@"a test" NSArray *myStrings = [[NSArray alloc] initWithObjects:string1, string2, string3,nil]; NSString *fullLengthString = [myStrings componentsJoinedByString:@" "]; 

or

you can use

NSString *imageFullName=[NSString stringWithFormat:@"%@ %@ %@.", string1,string2,string3]; 

Comments

1

Either of these formats work in XCode7 when I tested:

NSString *sTest1 = {@"This" " and that" " and one more"}; NSString *sTest2 = { @"This" " and that" " and one more" }; NSLog(@"\n%@\n\n%@",sTest1,sTest2); 

For some reason, you only need the @ operator character on the first string of the mix.

However, it doesn't work with variable insertion. For that, you can use this extremely simple solution with the exception of using a macro on "cat" instead of "and".

2 Comments

how to to make this? e.g: sTest3 = sTest1 + sTest2;
@user285594 That was the point of the question: that syntax is not allowed in Objective-C. See the other answers.
1

For all Objective C lovers that need this in a UI-Test:

-(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]; } 

Comments

0
listOfCatalogIDs =[@[@"id[]=",listOfCatalogIDs] componentsJoinedByString:@""]; 

Comments

0

Let's imagine that u don't know how many strings there.

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:@", "]; 

Comments

0

Inspired by NSMutableString idea from Chris, I make a perfect macro imho. It supports insert nil elements without any Exceptions.

#import <libextobjc/metamacros.h> #define STR_CONCAT(...) \ ({ \ __auto_type str__ = [NSMutableString string]; \ metamacro_foreach_cxt(never_use_immediately_str_concatify_,, str__, __VA_ARGS__) \ (NSString *)str__.copy; \ }) #define never_use_immediately_str_concatify_(INDEX, CONTEXT, VAR) \ [CONTEXT appendString:VAR ?: @""]; 

Example:

STR_CONCAT(@"button_bg_", @(count).stringValue, @".png"); // button_bg_2.png 

If you like, you can use id type as parameter by using [VAR description] instead of NSString.

Comments

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.