美文网首页iOS开发
Object-C单例写法2018

Object-C单例写法2018

作者: 勇往直前888 | 来源:发表于2018-09-30 14:10 被阅读40次

在项目中,单例用得比较多,当然代码基本上网上都有。大多数情况,也基本上自己写自己用,所以不会出现单例还用copy,alloc init之类的问题。
这里有一篇文章,感觉很不错,照着写了一下,基本上可以,当然稍有不同,所以备忘一下。

Objective-c单例模式的正确写法

调用接口

  • 名字可以统一为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;
  • copymutableCopy也需要重写一下,代码也是一样的
- (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] 来获取单例;

参考文章

下面两篇文章都不错,值得认真看看,推荐

相关文章

网友评论

    本文标题:Object-C单例写法2018

    本文链接:https://www.haomeiwen.com/subject/svtwoftx.html