This should be easy, but I\'m having a hard time finding the easiest solution.
I need an NSString
that is equal to another string concatenated with itse
The first method above is for a single character. This one is for a string of characters. It could be used for a single character too but has more overhead.
+ (NSString*)stringWithRepeatString:(char*)characters times:(unsigned int)repetitions;
{
unsigned int stringLength = strlen(characters);
unsigned int repeatStringLength = stringLength * repetitions + 1;
char repeatString[repeatStringLength];
for (unsigned int i = 0; i < repetitions; i++) {
unsigned int pointerPosition = i * repetitions;
memcpy(repeatString + pointerPosition, characters, stringLength);
}
// Set terminating null
repeatString[repeatStringLength - 1] = 0;
return [NSString stringWithCString:repeatString];
}
NSString *original = @"abc";
int times = 2;
// Capacity does not limit the length, it's just an initial capacity
NSMutableString *result = [NSMutableString stringWithCapacity:[original length] * times];
int i;
for (i = 0; i < times; i++)
[result appendString:original];
NSLog(@"result: %@", result); // prints "abcabc"
There is a method called stringByPaddingToLength:withString:startingAtIndex:
:
[@"" stringByPaddingToLength:100 withString: @"abc" startingAtIndex:0]
Note that if you want 3 abc's, than use 9 (3 * [@"abc" length]
) or create category like this:
@interface NSString (Repeat)
- (NSString *)repeatTimes:(NSUInteger)times;
@end
@implementation NSString (Repeat)
- (NSString *)repeatTimes:(NSUInteger)times {
return [@"" stringByPaddingToLength:times * [self length] withString:self startingAtIndex:0];
}
@end
If you're using Cocoa in Python, then you can just do that, as PyObjC imbues NSString
with all of the Python unicode
class's abilities.
Otherwise, there are two ways.
One is to create an array with the same string in it n
times, and use componentsJoinedByString:
. Something like this:
NSMutableArray *repetitions = [NSMutableArray arrayWithCapacity:n];
for (NSUInteger i = 0UL; i < n; ++i)
[repetitions addObject:inputString];
outputString = [repetitions componentsJoinedByString:@""];
The other way would be to start with an empty NSMutableString
and append the string to it n
times, like this:
NSMutableString *temp = [NSMutableString stringWithCapacity:[inputString length] * n];
for (NSUInteger i = 0UL; i < n; ++i)
[temp appendString:inputString];
outputString = [NSString stringWithString:temp];
You may be able to cut out the stringWithString:
call if it's OK for you to return a mutable string here. Otherwise, you probably should return an immutable string, and the stringWithString:
message here means you have two copies of the string in memory.
Therefore, I recommend the componentsJoinedByString:
solution.
[Edit: Borrowed idea to use …WithCapacity:
methods from Mike McMaster's answer.]
For performance, you could drop into C with something like this:
+ (NSString*)stringWithRepeatCharacter:(char)character times:(unsigned int)repetitions;
{
char repeatString[repetitions + 1];
memset(repeatString, character, repetitions);
// Set terminating null
repeatString[repetitions] = 0;
return [NSString stringWithCString:repeatString];
}
This could be written as a category extension on the NSString class. There are probably some checks that should be thrown in there, but this is the straight forward gist of it.