老生常谈之单例

本人有若干成套学习视频, 可试看! 可试看! 可试看, 重要的事情说三遍 包含Java, 数据结构与算法, iOS, 安卓, python, flutter等等, 如有需要, 联系微信tsaievan.

单例可以说是我们开发中最常用的设计模式之一了, 但如果面试的时候,人家让你手写单例, 你可以吗?

单例的设计思路

为什么要有单例这个东西呢?
如果我们想创建一个全局的, 唯一的变量, 这个时候就需要用到单例了
这个变量可能我们会在项目的各个地方使用, 但是我们不想占用太多的内存空间, 那么我们就会使用单例, 因为单例只会开辟一次内存空间

  • 全局只创建一次
    提供一个静态变量
static id _instanceType = nil;
  • 重写+ (instancetype)allocWithZone:(struct _NSZone *)zone方法, 因为我们在创建对象的时候使用的+ (instancetype)alloc方法, 最终还是会调用+ (instancetype)allocWithZone:(struct _NSZone *)zone方法
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
    if (!_instanceType) {
        _instanceType = [super allocWithZone:zone];
    }
    return _instanceType;
}

是不是有点像懒加载? 但这种写法是不严谨的, 当我们在子线程中创建对象的时候, 就可能造成多条线程同时访问+ (instancetype)allocWithZone:(struct _NSZone *)zone方法, 会造成线程不安全.

解决方案一: 加互斥锁
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
    @synchronized (self) {
        if (!_instanceType) {
            _instanceType = [super allocWithZone:zone];
        }
    }
    return _instanceType;
}
解决方案二: 使用GCD一次性代码
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        _instanceType = [super allocWithZone:zone];
    });
    return _instanceType;
}
注意: GCD一次性代码本身就是线程安全的
  • 暴露一个方法给外界使用
+ (instancetype)sharedInstance {
    return [[self alloc] init];
}
  • 重写- (id)copyWithZone:(NSZone *)zone方法和- (id)mutableCopyWithZone:(NSZone *)zone, 这样就比较严谨一点, 如果这个单例遵守了NSCopyingNSMutableCopying协议, 那么无论是使用copy方法还是mutableCopy方法, 都只会得到同一个实例对象
- (id)copyWithZone:(NSZone *)zone {
    return _instanceType;
}

- (id)mutableCopyWithZone:(NSZone *)zone {
    return _instanceType;
}
  • 你以为这样就结束了吗? 其实并没有, 倘若面试官问你在MRC下如何写单例的时候怎么办呢? 千万不能懵逼.
无非多了几个步骤
- 重写release方法
- 重写retain方法
- 重写retainCount方法

如果不重写这几个方法会怎样呢?

用不同的方式创建三个对象

当s1被release后, s1就已经被释放掉了, 这个时候单例对象指针指向的就是一个僵尸对象, 如下图:

访问僵尸对象报错

那么我们如何让单例对象不被释放掉了, 那就是要release方法失灵, 那么重写release方法, 就什么都不做. retain方法类似, 也什么都不做, 只返回一个当前的单例对象出去, retainCount方法就返回一个最大值出去.

- (oneway void)release {
    
}

- (instancetype)retain {
    return _instanceType;
}

- (NSUInteger)retainCount {
    return MAXFLOAT;
}

此时的单例应该就完成得差不多了, 但是还有一个问题, 单例能够继承吗? 在java等其他语言中, 单例是不可以继承的, 但在OC中, 虽然你可以那么做, 但实际的效果是, 要么你得到的全是父类, 要么全是子类, 完全不符合我们的要求, 那么, 如果我不想创建一个单例就写上面一串代码,想一劳永逸怎么办呢?

答案是: 创建一个单例宏

而且我们可以写一个条件编译的代码, 这样无论是ARC环境还是MRC环境, 都可以用, 我们需要用到这样的条件编译指令

#if __has_feature(onjc_arc)

#else

#endif

我们还可以创建一个带参数的宏, 这样就能自定义单例的构造方法

#define SINGLETON_INTERFACE(NAME) + (instancetype)shared##NAME;

#if __has_feature(onjc_arc)

#define SINGLETON_IMPLEMENTATION(NAME) static id _instanceType = nil;\
\
+ (instancetype)allocWithZone:(struct _NSZone *)zone {\
    static dispatch_once_t onceToken;\
    dispatch_once(&onceToken, ^{\
        _instanceType = [super allocWithZone:zone];\
    });\
    return _instanceType;\
}\
\
+ (instancetype)shared##NAME {\
    return [[self alloc] init];\
}\
\
- (id)copyWithZone:(NSZone *)zone {\
    return _instanceType;\
}\
\
- (id)mutableCopyWithZone:(NSZone *)zone {\
    return _instanceType;\
}\

#else

#define SINGLETON_IMPLEMENTATION(NAME) static id _instanceType = nil;\
\
\
+ (instancetype)allocWithZone:(struct _NSZone *)zone {\
    static dispatch_once_t onceToken;\
    dispatch_once(&onceToken, ^{\
        _instanceType = [super allocWithZone:zone];\
    });\
    return _instanceType;\
}\
\
+ (instancetype)shared##NAME {\
    return [[self alloc] init];\
}\
\
- (id)copyWithZone:(NSZone *)zone {\
    return _instanceType;\
}\
\
- (id)mutableCopyWithZone:(NSZone *)zone {\
    return _instanceType;\
}\
\
- (oneway void)release {\
\
}\
\
- (instancetype)retain {\
    return _instanceType;\
}\
\
- (NSUInteger)retainCount {\
    return MAXFLOAT;\
}\
\

#endif
这样我们在创建单例的时候就非常方便了, 直接把这个.h文件拖进去, 写两个宏就OK了

PS. 本人有若干成套学习视频, 包含Java, 数据结构与算法, iOS, 安卓, python, flutter等等, 如有需要, 联系微信tsaievan.

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

推荐阅读更多精彩内容

  • 单例模式的作用 可以保证在程序运行过程,一个类只有一个实例,而且该实例易于供外界访问 从而方便地控制了实例个数,并...
    JonesCxy阅读 380评论 0 0
  • 一. ARC环境下的单例模式 单例模式的基本概念单例, 顾名思义, 即在整个程序中, 某一个类只有唯一一个实例, ...
    面糊阅读 754评论 0 50
  • iOS-单例模式写一次就够了 一. 单例模式简介 单例模式的作用 可以保证在程序运行过程,一个类只有一个实例,而且...
    Moker_C阅读 392评论 0 1
  • 什么是单例模式? >是开发设计模式(共23种)中的1种 >它可以保证在程序运行过程,一个类只有一个实例(一个对象)...
    泥孩儿0107阅读 244评论 0 0
  • 单例模式的作用 可以保证在程序运行过程,一个类只有一个实例,而且该实例易于供外界访问 从而方便地控制了实例个数,并...
    喝完酒再来杯拉菲阅读 147评论 0 1