Objective-C实现单例

单例,顾名思义,即有且只有一个实例。

例如我们用到的NSNotificationCenter、NSFileManager等,其default方法,就是用的单例技术。

那么创建函数(sharedInstance)被多次调用,以及多线程场景下,如何保证这一点?

一、从苹果的一些文档中发现官方给出的方式是:

//在.m中实现如下内容及方法

static MyClass *class = nil; //定义静态全局变量

@implementation MyClass
+(MyClass *)sharedMyClass{
    @synchronized(self){  //为了确保多线程情况下,仍然确保实体的唯一性
        if (!class) {
            [[self alloc] init]; //该方法会调用 allocWithZone
        }
    }
    return class;
}

// 重写alloc,确保使用同一块内存
+(id)allocWithZone:(NSZone *)zone{
    @synchronized(self){
        if (!class) {
            class = [super allocWithZone:zone]; //确保使用同一块内存地址
            return class;
        }
    }
    return nil;
}

- (id)copyWithZone:(NSZone *)zone;{
    return self; //确保copy对象也是唯一
}

-(id)retain{
    return self; //确保计数唯一
}

- (unsigned)retainCount
{
   return UINT_MAX;  //计数永远为-1
}

- (id)autorelease
{
    return self;//确保自动计数唯一
} 

- (oneway void)release
{
     //重写计数释放方法
}

@end

这种方法真正的确保了使用同一块内存块,确保了object的唯一性。

唯一的缺点是这种方法实现全起来有点麻烦。

 

二、使用GCD技术:

从引入Grand Central Dispatch (GCD)后, 一般使用GCD中的dispatch_once函数来实现。

函数形式如下:

void dispatch_once( dispatch_once_t *predicate, dispatch_block_t block);

第一个参数predicate,该参数是检查后面第二个参数所代表的代码块是否被调用的谓词,

第二个参数则是在整个应用程序中只会被调用一次的代码块。

dispach_once函数可以保证不管被调用几次,函数中的代码块(block)只会被执行一次,而且还是线程安全的。

+(instancetype)sharedInstance
{
    static ClassName* _sharedInstance = nil;
    static dispatch_once_t oncePredicate;
    dispatch_once (&oncePredicate, ^{
        _sharedInstance = [[ClassName alloc] init];
    });
    return _sharedInstance;
}

这里充分利用了static的特性:

static变量只会被初始化一次,之后的函数调用会使用变量上一次的值,而不会重新创建一个。

_sharedInstance和oncePredicate都只有一个。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

auspark

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值