在项目中,单例用得比较多,当然代码基本上网上都有。大多数情况,也基本上自己写自己用,所以不会出现单例还用copy,alloc init
之类的问题。
这里有一篇文章,感觉很不错,照着写了一下,基本上可以,当然稍有不同,所以备忘一下。
调用接口
- 名字可以统一为
sharedInstance
,放在h
文件中:
+ (instancetype)sharedInstance;
-
静态常量,不需要全局的,写成局部变量就可以。
static
类型,全局和局部差别不大,所以写成局部变量就可以。 -
初始化用
allocWithZone
,不用alloc
; -
单例的类型,不需要指定具体的类,只需要通用的
id
就可以了。
+ (instancetype)sharedInstance {
static id instance = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
instance = [[super allocWithZone:NULL] init];
});
return instance;
}
重写方法
-
重写
allocWithZone
,让alloc init
也返回单例。 -
不用出现具体的类名,用
[self class]
替代。
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
return [[self class] sharedInstance];
}
- 在
ARC
下,没有copyWithZone
函数,这个可以从NSObject.h
中看到
+ (id)copyWithZone:(struct _NSZone *)zone OBJC_ARC_UNAVAILABLE;
+ (id)mutableCopyWithZone:(struct _NSZone *)zone OBJC_ARC_UNAVAILABLE;
-
copy
和mutableCopy
也需要重写一下,代码也是一样的
- (id)copy {
return [[self class] sharedInstance];
}
- (id)mutableCopy {
return [[self class] sharedInstance];
}
例子代码
这样一来,所有的单例就可以用完全相同的代码来表示了:
+ (instancetype)sharedInstance {
static id instance = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
instance = [[super allocWithZone:NULL] init];
});
return instance;
}
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
return [[self class] sharedInstance];
}
- (id)copy {
return [[self class] sharedInstance];
}
- (id)mutableCopy {
return [[self class] sharedInstance];
}
测试代码
- 使用的时候用类名,也就是具体的文件名字。
KJTWebViewManager* obj1 = [KJTWebViewManager sharedInstance] ;
NSLog(@"obj1 = %@.", obj1);
KJTWebViewManager* obj2 = [KJTWebViewManager sharedInstance] ;
NSLog(@"obj2 = %@.", obj2);
KJTWebViewManager* obj3 = [[KJTWebViewManager alloc] init];
NSLog(@"obj3 = %@.", obj3);
KJTWebViewManager* obj4 = [[KJTWebViewManager alloc] init];
NSLog(@"obj4 = %@.", [obj4 copy]);
KJTWebViewManager* obj5 = [[KJTWebViewManager alloc] init];
NSLog(@"obj4 = %@.", [obj5 mutableCopy]);
- 输出结果,所有对象的地址都是一样的,也不会崩溃。如果没有重写
copy
或者mutableCopy
,用到的时候会崩溃。
2018-09-30 13:38:12.659190+0800 Wallet[42247:11822152] +[KJTWebView showWithController:url:] 第18行
obj1 = <KJTWebViewManager: 0x600001ab1630>.
2018-09-30 13:38:12.659417+0800 Wallet[42247:11822152] +[KJTWebView showWithController:url:] 第20行
obj2 = <KJTWebViewManager: 0x600001ab1630>.
2018-09-30 13:38:12.659550+0800 Wallet[42247:11822152] +[KJTWebView showWithController:url:] 第22行
obj3 = <KJTWebViewManager: 0x600001ab1630>.
2018-09-30 13:38:12.659635+0800 Wallet[42247:11822152] +[KJTWebView showWithController:url:] 第24行
obj4 = <KJTWebViewManager: 0x600001ab1630>.
2018-09-30 13:38:14.272292+0800 Wallet[42247:11822152] +[KJTWebView showWithController:url:] 第26行
obj5 = <KJTWebViewManager: 0x600001ab1630>.
宏替换
既然所有的单例都完全一样的代码,那么就考虑用宏来进行统一的替换。
- 头文件中的接口名字要对起来,所以要提供一个单例的
interface
#define kSingletonInterface + (instancetype)sharedInstance;
- 实现文件中就是上面贴的所有单例完全一样的样例代码:
#define kSingletonImplementation \
\
+ (instancetype)sharedInstance { \
static id instance = nil; \
static dispatch_once_t onceToken; \
dispatch_once(&onceToken, ^{ \
instance = [[super allocWithZone:NULL] init]; \
}); \
return instance; \
} \
\
+ (instancetype)allocWithZone:(struct _NSZone *)zone { \
return [[self class] sharedInstance]; \
} \
\
- (id)copy { \
return [[self class] sharedInstance]; \
} \
\
- (id)mutableCopy { \
return [[self class] sharedInstance]; \
}
使用宏
- 头文件中是这样的:
@interface KJTWebViewManager : NSObject
kSingletonInterface
@end
- 实现文件中是这样的:
#import "KJTWebViewManager.h"
@implementation KJTWebViewManager
kSingletonImplementation
@end
-
不同单例,只要换不同的类名和文件名就可以了。
-
使用单例还是像上面的测试代码一样,用
[类名 sharedInstance]
来获取单例;
参考文章
下面两篇文章都不错,值得认真看看,推荐
网友评论