单例应用场合
在iOS的生命周期内,有时候我们只需要某个类的一个实例。
例如UIApplication, UIAccelerometer, NSUserDefaults, NSNotificationCenter, NSFileManger, NSBundle这些都是单例类的典型代表。
如何创建单例类
- 创建一个静态对象
- 提供一个静态方法
- 重写alloc方法
代码演示
我们首先创建一个类Singleton, 继承自NSObject.
添加一个静态全局变量:
static Singleton *_singleton = nil;
创建一个静态的方法:
+(Singleton *)shareSingleton { //返回对象前需要判断,这个对象之前是否创建过,如果没有创建过,就需要创建一个对象,如果创建过,就把上一次的对象返回出去。这样可以保证创建的对象是唯一的对象 //@synchronized:多个线程同时访问单例类,就会创建多个单例类,我们需要加锁。 @synchronized(self) { if (_singleton == nil) { _singleton = [[self alloc]init]; } return _singleton; } }
我们创建单例类的目的是在全局只能创建一个对象,但是如果我们使用上面的方法创建有一个问题,就是如果用户不使用shareSingleton创建对象,而是使用alloc方法创建对象,就会又生成一个对象实例,这跟我们项目中只生成一个冲突,那么我们怎么办呢?
第一反应,我们需要重写alloc方法,实际上我们没有必要重写alloc方法。(下文会详述)
我们先重写alloc方法:
+(id)alloc { @synchronized(self) { if (_singleton == nil) { _singleton = [super alloc]; } return _singleton; } }
既然用户可能用alloc方法创建对象,那么用户有没有可能使用allocWithZone创建对象呢?
我们在此又需要重写allocWithZone这个方法:
+(id)allocWithZone:(struct _NSZone *)zone { @synchronized(self) { if (_singleton == nil) { _singleton = [super allocWithZone:zone]; } return _singleton; } }
重写alloc,和重写allocWithZone有什么区别呢?
allocWithZone是一个比较全面的方法,在alloc方法内部,会调用allocWithZone这个方法。
使用allocWithZone不会调用alloc
所以没必要重写alloc,直接重写allocWithZone即可。
基本的代码是:
#import "Singleton.h" static Singleton *_singleton = nil; @implementation Singleton +(Singleton *)shareSingleton { @synchronized(self) { if (_singleton == nil) { _singleton = [[self alloc]init]; } return _singleton; } } +(id)allocWithZone:(struct _NSZone *)zone { @synchronized(self) { if (_singleton == nil) { _singleton = [super allocWithZone:zone]; } return _singleton; } }
上述的代码只是在ARC中可以正确使用,如果实在MRC中,有retain,copy,release, autorelease,这些方法都会使得引用计数变化,我们都需要对这些方法重写。
-(id)retain { return _singleton; } -(void)release { } -(id)autorelease { return _singleton; } -(id)copy { return _singleton; } -(NSUInteger)retainCount { return 1; }