Предположим, у меня есть NSDictionary с двумя вложенными коллекциями NSArray и NSDictionary:
NSMutableDictionary *mkDict(void){
NSMutableDictionary *dict=[NSMutableDictionary dictionary];
NSMutableDictionary *sub=[NSMutableDictionary dictionary];
NSMutableArray *array= [NSMutableArray array];
[dict setObject:array forKey:@"array_key"];
[dict setObject:sub forKey:@"dict_key"];
return dict;
}
Существует множество способов получить доступ к одному элементу вложенной коллекции, и я выбрал время для трех из них.
Первый способ - это косвенный доступ к подэлементам путем доступа к ключу родительского элемента:
void KVC1(NSMutableDictionary *dict, int count){
for(int i=0; i<count; i++){
char buf1[40], buf2[sizeof buf1];
snprintf(buf1,sizeof(buf1),"element %i", i);
snprintf(buf2, sizeof buf2, "key %i", i);
[[dict objectForKey:@"array_key"]
addObject:
[NSString stringWithUTF8String:buf1]];
[[dict objectForKey:@"dict_key"]
setObject:[NSString stringWithUTF8String:buf1]
forKey:[NSString stringWithUTF8String:buf2]];
}
}
Второй - использовать KeyPath-доступ:
void KVC2(NSMutableDictionary *dict, int count){
for(int i=0; i<count; i++){
char buf1[40], buf2[sizeof buf1], buf3[sizeof buf1];
snprintf(buf1,sizeof(buf1),"element %i", i);
snprintf(buf2, sizeof buf2, "key %i", i);
snprintf(buf3, sizeof buf3, "dict_key.key %i",i);
[dict insertValue:
[NSString stringWithUTF8String:buf1]
atIndex:i inPropertyWithKey:@"array_key"];
[dict setValue:
[NSString stringWithUTF8String:buf1]
forKeyPath:
[NSString stringWithUTF8String:buf3]];
}
}
И третий, аналогичный первому, - получить доступ к указателю на подэлемент, а затем использовать этот указатель:
void KVC3(NSMutableDictionary *dict, int count){
NSMutableArray *subArray = [dict objectForKey:@"array_key"];
NSMutableDictionary *subDict = [dict objectForKey:@"dict_key"];
for(int i=0; i<count; i++){
char buf1[40], buf2[sizeof buf1];
snprintf(buf1,sizeof(buf1),"element %i", i);
snprintf(buf2, sizeof buf2, "key %i", i);
[subArray addObject:[NSString stringWithUTF8String:buf1]];
[subDict
setObject:
[NSString stringWithUTF8String:buf1]
forKey:
[NSString stringWithUTF8String:buf2]];
}
}
Вот код времени:
#import <Foundation/Foundation.h>
#import <mach/mach_time.h>
// KVC1, KVC2 and KVC3 from above...
#define TIME_THIS(func,times) \
({\
mach_timebase_info_data_t info; \
mach_timebase_info(&info); \
uint64_t start = mach_absolute_time(); \
for(int i=0; i<(int)times; i++) \
func ; \
uint64_t duration = mach_absolute_time() - start; \
duration *= info.numer; \
duration /= info.denom; \
duration /= 1000000; \
NSLog(@"%i executions of line %i took %lld milliseconds", times, __LINE__, duration); \
});
int main (int argc, const char * argv[]) {
NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
NSMutableDictionary *dict=mkDict();
NSMutableDictionary *dict2=mkDict();
NSMutableDictionary *dict3=mkDict();
TIME_THIS(KVC1(dict,1000),10);
TIME_THIS(KVC2(dict2,1000),10);
TIME_THIS(KVC3(dict3,1000),10);
if([dict isEqualToDictionary:dict2])
NSLog(@"And they are the same...");
[pool drain];
return 0;
}
Вот результаты:
10 executions of line 256 took 57 milliseconds
10 executions of line 257 took 7930 milliseconds
10 executions of line 258 took 46 milliseconds
And they are the same...
Вопрос: Почему OS X Snow Leopard / Lion предложил метод использования KeyPaths настолько вонючий медленный? Если вы увеличите размер count
до 10000 или более, KVC2 станет бесконечно медленным, тогда как два других метода увеличиваются линейно.
Я что-то не так делаю? Есть ли лучшая идиома для доступа к одному элементу вложенной коллекции в словаре?