Objective-C 单例模式的实现

本文探讨了Objective-C中实现单例模式时的线程安全问题,指出最简单写法可能导致多线程环境下创建多个实例。并介绍了使用`dispatch_once`来确保单例的线程安全性,保证在程序运行期间block只执行一次,从而避免了并发问题导致的异常。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

OC中实现单例模式

最简单的写法,存在线程安全问题:

@implementation Singleton

+ (instancetype)shareInstance {
    static Singleton *instance = nil;
    if (! instance) {
        instance = [[Singleton alloc] init];
    }
    return instance;
}

@end

存在的线程安全问题是,可能出现两条线程同时判断了instance==nil,然后实例了两个Singleton出来。

测试代码:

dispatch_queue_t queue = dispatch_queue_create("gcd_test_label", DISPATCH_QUEUE_CONCURRENT);
    for (int i=0; i<10; i++) {
        dispatch_async(queue, ^(void) {
            Singleton *singleton = [Singleton shareInstance];
        });
    }

运行结果:
在这里插入图片描述
甚至出现了EXC_BAD_ACCESS中断。

以下介绍几种实现线程安全的单利模式。

使用dispatch_once

//Singleton.h
@interface Singleton: NSObject

+ (instancetype)shareInstance;

//封锁init方法
- (instancetype)init NS_UNAVAILABLE;

//封锁new方法
+ (instancetype)new NS_UNAVAILABLE;

//封锁alloc方法
+ (instancetype)alloc NS_UNAVAILABLE;

@end

//Singleton.m
@implementation Singleton

+ (instancetype)shareInstance {
    static Singleton *instance = nil;
    static dispatch_once_t predicate;

    dispatch_once(&predicate, ^(void) {
        instance = [[super allocWithZone:nil] initInner];
    });
    return instance;
}

+ (instancetype)allocWithZone:(struct _NSZone *)zone {
    return [Singleton shareInstance];
}

- (instancetype)initInner
{
    self = [super init];
    if (self) {
        NSLog(@"Singleton init.");
    }
    return self;
}

@end

dispatch_once 保证了在predicate相同的条件下,传入的block在程序生命期内只执行一次。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值