iOS陷阱--NSDateFormatter的线程安全性

经常需要将NSDate和NSString进行互转,一般我们会这么写:
  1. NSDate转NSString
    NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
    [dateFormatter setDateFormat:@”yyyy-MM-dd’T’HH:mm:ss”];

    [dateFormatter setTimeZone:[NSTimeZone timeZoneWithName:@”Asia/Shanghai”]];
    return [dateFormatter dateFromString:dateString];
  2. NSString转NSDate
    NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
    NSTimeZone *timeZone = [NSTimeZone defaultTimeZone];
    [dateFormatter setTimeZone:timeZone];
    [dateFormatter setDateFormat:@”yyyy-MM-dd HH:mm:ss Z”];
    return [dateFormatter stringFromDate:date];
    上面运行起来好像很顺畅,可是有一天,老大跑过来问:“hey,小张,有用户反应我们的App滑动很卡啊”。于是你打开Profile工具查了一下性能,你会发现上述两个转换函数占CPU比例非常高。怎么优化呢?
  1. 延迟转换并Cache
    即只有在UI需要使用转换结果时再执行转换,并将结果缓存起来。
  2. NSDateFormatter对象只生成1次
   方法1很好容易实现, 下面我们实现方法2:
static NSDateFormatter *dateFormatter = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
    dateFormatter = [[NSDateFormatter alloc] init];
});
[dateFormatter setDateFormat:@”yyyy-MM-dd’T’HH:mm:ss”];
[dateFormatter setTimeZone:[NSTimeZone timeZoneWithName:@”Asia/Shanghai”]];
return [dateFormatter dateFromString:dateString];

    貌似也挺容易的,哈哈经过上面的改造后,你会发现性能得到了显著提升,非常高兴的给老大发布新版本,很快老大又过来把你劈头盖脸地训了一顿:“怎么老是Crash,这么不稳定?赶紧fix”。
    一查CrashReport,发现在[dateFormatter dateFromString:dateString]这里出了问题。这就奇怪了?系统函数库也有bug么?
    当然不是了,这是因为 NSDateFormatter不是线程安全的 ,请查看 https://developer.apple.com/library/mac/documentation/cocoa/conceptual/Multithreading/ThreadSafetySummary/ThreadSafetySummary.html ,里面列出了线程安全和不安全的类。当多个线程同时访问1个NSDateFormatter对象时,有可能会Crash。
那怎么办?不要泄气,咱们给每个线程准备1个NSDateFormatter,大家就不会打得头破血流了。
解决方案:
NSMutableDictionary *threadDictionary = [[NSThread currentThread] threadDictionary];
NSDateFormatter *dateFormatter = threadDictionary[@”mydateformatter”];

if(!dateFormatter){
    @synchronized(self){
        if(!dateFormatter){
            dateFormatter = [[NSDateFormatter alloc] init];
           [dateFormatter setDateFormat:@”yyyy-MM-dd HH:mm:ss”];
           [dateFormatter setTimeZone:[NSTimeZone timeZoneWithName:@”Asia/Shanghai”]];
          threadDictionary[@”mydateformatter”] = dateFormatter;
         }
    }
}
发现没有,上面使用了 double-check 技术哦,哈哈,这里千万不要再用dispatch_once哈。

欢迎关注“iOS开发之道”的微博和微信帐号,一起交流学习。
微博:iOS开发之道  http://weibo.com/u/3652772421
微信:ioszhidao

你可能感兴趣的:(iOS开发)