单例模式只能修改无法释放,直到程序结束。
我们下面一步一步来做一个单例模式程序
(1)单例一旦创建,是永远存在于内存中的,所以需要创建一个全局量
static MySingletonClass *sharedSingleTonObj=nil;
(2)既然是单例,一定有一个构造方法直接忽略跳过实例对象的生成过程。据此看来“类方法”最合适不过了
+(MySingletonClass *)sharedSingleton
{
//多线程安全的关键字,相关概念可以参考多线程编程章节
@synchronized(self)
{
//创建
if(sharedSingletonObj==nil)
{
sharedSingletonObj= [[super allocWithZone:NULL] init];
}
//发出必要警告
else
{
NSLog(@"单例对象已经存在!");
}
}
return sharedSingletonObj;
}
(3)如果对sharedSingletonObj执行了copy呢?我们需要重新写copy方法
-(id)copyWithZone:(NSZone *)zone
{
return self;
}
(4)如果对sharedSingletonObj执行了retain呢?我们同样需要重写retain方法。
-(id)retain
{
return self;
}
(5)继续对release和autorelease的方法重写
-(void)release
{
} -(id)autorelease
{
return self;
}
(6)重要的一点!我们需要实现NSObjec里面,关于引用计数API的重写以避免因为引用计数为0导致dealloc的触发
-(NSUInteger)retainCount
{
//是一个无限大的int数,避免额系统自动触发单例dealloc方法
//也可以明确告知调用者,这是一个单例模式
return NSUIntegerMax;
}
(7)最后,你会发现就这样让人使用的话,如果不通过类方法创建对象转而调用alloc创建,则每次会分配新内存引用计数+1,显然alloc方法势必也不要重写
+(id)allocWithZone:(NSZone *)zone
{
//直接套用sharedSingleton,retain 符合alloc惯例
//使类方法返回的对象的引用计数+1,此处retain根据上面的重写内容,不做任何事情
return [[MySingletonClass sharedSingleton] retain];
}
我们需要注意的时:往往只需要一个单例对象而已,但是如果仍相用alloc和init创建这个类的其他对象,那上述写法中,我们不需要重写2,3,4,5,6的代码
注:如果用gcd的话 ,只要在.m文件中实现这就可以 .h文件中声明一下这个 + (XEMessageData*) sharedInstance;
+ (XEMessageData*) sharedInstance
{
static XEMessageData* sharedInstance;
static dispatch_once_t predicate;
dispatch_once(&predicate,^{
sharedInstance = [[self alloc] init];
sharedInstance.tempConversationMessageID2ConversationID = [[NSMutableDictionary alloc] init];
}); return sharedInstance;
}