iOS-严谨的单例设计模式

什么是单例模式

  • 单例模式就是要保证系统中一个类只有一个对象实例。无论用什么方法创建多少次,所得的对象都是同一个对象。

单例模式的应用场景

  • 在iOS开发中,我们已经遇到过很多单例模式的身影:
    • [UIApplication sharedApplication]、[NSUserDefaults standardUserDefaults]、[NSNotificationCenter defaultCenter] 等等。
    • 音乐播放器中用于播放音乐的播放器对象、一个APP中用于保存并且能够随时方便地获取的用户信息的对象 等等。

单例模式的关键

  1. 对象只创建一次

  2. 可供全局访问

  3. 不会被释放,直至程序结束

单例模式的分析与实现

  • 对象只创建一次:
    在iOS中我们创建一个对象一般用:alloc init 或 new,其中new方法内部实际也是通过alloc init创建的,所以我们把注意力放在alloc init上。首先alloc方法是给对象分配内存空间,然后init方法是对该对象进行初始化,所以想要控制对象的创建关键是在alloc方法上,又由于alloc默认是调用allocWithZone方法,所以我们应该重写allocWithZone方法来控制对象只能创建一次:
id instance; // 定义全局变量来保存单例对象,此处还不完善,后面还会提到
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
    // 此处if判断可以避免频繁加锁,只要对象已创建就直接返回,亦相当于懒加载
    if (instance == nil) {
        // 方法一:互斥锁方式
        @synchronized(self) {
            if (instance == nil) {
                instance = [super allocWithZone:zone]; // 用super去调用,避免死循环
            }
        }
        // 方法二:GCD一次性代码方式
        static dispatch_once_t onceToken;
        dispatch_once(&onceToken, ^{
            instance = [super allocWithZone:zone]; // 用super去调用,避免死循环
        });
    }
    return instance;
}
  • 可供全局访问:
    单例模式要提供一个类方法来获取单例对象,例如:

    Tools *tools = [Tools sharedTools];
    UserTool *userTool = [UserTool defaultUserTool];
    

    实现如下:

    // 单例类方法 命名规则: shared + 类名 或 default + 类名
    + (instancetype)sharedTools {
        if (instance == nil) {
            instance = [self alloc] init]; // 最终还是调用allocWithZone方法
        }
        return instance;
    }
    
  • 不会被释放,直至程序结束:
    在第一个关键点中,我们定义了一个全局变量 id instance;来保存我们创建的单例对象,但是有个弊端,如果在别的文件中(别的类)使用extern关键字来获取这个对象是可以拿到的,并且可以把该对象销毁,例如:

    extern id instance;
    instance = nil;
    

    这样以来,下次再获取单例对象的时候发现为nil就会重新创建对象,即二次创建对象,亦即不为单例模式,为了防止单例对象的销毁,我们应该使用static修饰用于保存单例对象的变量,限制变量的作用域为此文件可用,那么别的文件(别的类)就无法拿到这个对象,从而达到单例对象不会被释放。
    即把id instance;改为static id instance;

严谨的单例模式

  • 创建对象除了alloc init 和 new 以外,还可以通过copy 和 mutableCopy来创建对象,为了严谨起见,我们还需要控制这两个方法的创建过程,即需要重写copyWithZone和mutableCopyWithZone方法,重写这两个方法需要分别遵守NSCopying 和 NSMutableCopying协议。
    因为这两个方法是对象方法,所以当想要使用这两个方法来创建新对象的时候,只能是用单例对象来调用此方法,即单例对象已经创建了,所以我们只需要返回我们保存的单例对象即可,代码如下:

    - (id)copyWithZone:(NSZone *)zone {
        return instance;
    }
    - (id)mutableCopyWithZone:(NSZone *)zone {
        return instance;
    }
    

至此一个严谨的单例设计模式已经完成了,下面附上完整代码:

#import "Tools.h"

@implementation Tools

static id instance;
+ (instancetype)sharedTools {
    if (instance == nil) {
        instance = [[self alloc] init];
    }
    return instance;
}
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
    // 此处if判断可以避免频繁加锁,只要对象已创建就直接返回,亦相当于懒加载
    if (instance == nil) {
        // 方法一:互斥锁方式
        @synchronized(self) {
            if (instance == nil) {
                instance = [super allocWithZone:zone]; // 用super去调用,避免死循环
            }
        }
        // 方法二:GCD一次性代码方式
        static dispatch_once_t onceToken;
        dispatch_once(&onceToken, ^{
            instance = [super allocWithZone:zone]; // 用super去调用,避免死循环
        });        
    }
    return instance;
}
// 遵守NSCopying协议 
- (id)copyWithZone:(NSZone *)zone {
    return instance;
}
// 遵守NSMutableCopying协议
- (id)mutableCopyWithZone:(NSZone *)zone {
    return instance;
}
@end

参考文章

http://lib.csdn.net/article/ios/35938

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,294评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,493评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,790评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,595评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,718评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,906评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,053评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,797评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,250评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,570评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,711评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,388评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,018评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,796评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,023评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,461评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,595评论 2 350

推荐阅读更多精彩内容