从实例到单例:Objective-C 单例类的线程安全实现方案

从实例到单例:Objective-C 单例类的线程安全实现方案

在 Objective-C 中实现线程安全的单例模式需解决两个核心问题:

  1. 确保全局唯一实例
  2. 防止多线程环境下的竞态条件
一、基础实现方案
objectivec 复制代码
// MySingleton.h
@interface MySingleton : NSObject
+ (instancetype)sharedInstance;
@end

// MySingleton.m
@implementation MySingleton
+ (instancetype)sharedInstance {
    static MySingleton *_sharedInstance = nil;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        _sharedInstance = [[self alloc] init];
    });
    return _sharedInstance;
}
@end

关键组件

  • static 变量保证实例生命周期与程序相同
  • dispatch_once_t 确保代码块仅执行一次
  • GCD 的 dispatch_once 提供原子性操作
二、线程安全分析

T_1, T_2 为并发线程,其执行过程满足: $$P(T_1 \cap T_2) = \emptyset \implies \text{单例唯一性}$$ dispatch_once 的底层实现基于:

  1. 原子性内存屏障
  2. 信号量锁机制
  3. 双重检查锁定模式
三、防御性改进方案

防止通过其他方式创建实例:

objectivec 复制代码
// 重写 allocWithZone 方法
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
    return [self sharedInstance];
}

// 防止拷贝
- (id)copyWithZone:(NSZone *)zone {
    return self;
}
四、性能对比
实现方案 线程安全 时间复杂度 内存屏障
@synchronized O(n)
dispatch_once O(1)
双重检查锁定 O(1)

最佳实践

  1. 始终使用 dispatch_once 方案
  2. .m 文件中声明静态变量而非头文件
  3. 重写 allocWithZone:copyWithZone: 方法
  4. 避免在单例初始化方法中调用 sharedInstance
五、使用场景

适合以下全局服务:

  • 配置管理器
  • 网络监控器
  • 核心数据栈
  • 日志系统
graph LR A[调用 sharedInstance] --> B{实例是否存在?} B -- 否 --> C[dispatch_once 创建] B -- 是 --> D[返回现存实例] C --> D

此方案满足: $$ \forall t \in \mathbb{T}, \quad \exists! , \text{instance} \in \text{MySingleton} $$ 其中 \\mathbb{T} 表示线程集合,\\exists! 表示存在且唯一。

相关推荐
Wenweno0o1 天前
0基础Go语言Eino框架智能体实战-chatModel
开发语言·后端·golang
chenjingming6661 天前
jmeter线程组设置以及串行和并行设置
java·开发语言·jmeter
cch89181 天前
Python主流框架全解析
开发语言·python
不爱吃炸鸡柳1 天前
C++ STL list 超详细解析:从接口使用到模拟实现
开发语言·c++·list
十五年专注C++开发1 天前
RTTR: 一款MIT 协议开源的 C++ 运行时反射库
开发语言·c++·反射
Momentary_SixthSense1 天前
设计模式之工厂模式
java·开发语言·设计模式
‎ദ്ദിᵔ.˛.ᵔ₎1 天前
STL 栈 队列
开发语言·c++
勿忘,瞬间1 天前
数据结构—顺序表
java·开发语言
张張4081 天前
(域格)环境搭建和编译
c语言·开发语言·python·ai
weixin_423533991 天前
【Windows11离线安装anaconda、python、vscode】
开发语言·vscode·python