我就直接用个例子记录:
工程代码记录下来:
//
// Ticket.h
#import <Foundation/Foundation.h>
@interface Ticket : NSObject
// 实例化票据的单例
+ (Ticket *)sharedTicket;
// 在多线程应用中,所有被抢夺资源的属性需要设置为原子属性
// 系统会在多线程抢夺时,保证该属性有且仅有一个线程能够访问
// 注意:使用atomic属性,会降低系统性能,在开发多线程应用时,尽量不要资源
// 另外,atomic属性,必须与@synchronized(同步锁)一起使用
// 票数
@property (assign, atomic) NSInteger tickets;
@end
//
// Ticket.m
/**
实现单例模型需要做三件事情
1. 使用全局静态变量记录住第一个被实例化的对象
static Ticket *SharedInstance
2. 重写allocWithZone方法,并使用dispatch_once_t,从而保证在多线程情况下,
同样只能实例化一个对象副本
3. 建立一个以shared开头的类方法实例化单例对象,便于其他类调用,同时不容易引起歧义
同样用dispatch_once_t确保只有一个副本被建立。
另外关于被抢夺资源使用的注意事项
在多线程应用中,所有被抢夺资源的属性需要设置为原子属性
系统会在多线程抢夺时,保证该属性有且仅有一个线程能够访问
注意:使用atomic属性,会降低系统性能,在开发多线程应用时,尽量不要资源
另外,atomic属性,必须与@synchronized(同步锁)一起使用
*/
#import "Ticket.h"
static Ticket *SharedInstance;//使用全局静态变量记录住第一个被实例化的对象
@implementation Ticket
// 使用内存地址实例化对象,所有实例化方法,最终都会调用此方法
// 要实例化出来唯一的对象,需要一个变量记录住第一个实例化出来的对象
+ (id)allocWithZone:(NSZone *)zone
{
// 解决多线程中,同样只能实例化出一个对象副本
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
SharedInstance = [super allocWithZone:zone];
});
return SharedInstance;
}
// 建立一个单例对象,便于其他类调用
+ (Ticket *)sharedTicket
{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
SharedInstance = [[Ticket alloc]init];
});
return SharedInstance;
}
@end
//
// MainViewController.m文件
//
#import "MainViewController.h"
#import "Ticket.h"
- (void)viewDidLoad
{
[super viewDidLoad];
[Ticket sharedTicket].tickets = 30;//单例使用
}
创建一个单例很多办法。我先列举一个苹果官方文档中的写法。
当然,在iOS4之后有了另外一种写法:
该写法来自 objcolumnist,文中提到,该写法具有以下几个特性:
1. 线程安全。
2. 满足静态分析器的要求。
3. 兼容了ARC
然后我还有点好奇的是dispatch_once,这个函数,没见过啊。
于是就到官方的文档里找找看,是怎么说的。
下面是官方文档介绍:
该方法的作用就是执行且在整个程序的声明周期中,仅执行一次某一个block对象。简直就是为单例而生的嘛。而且,有些我们需要在程序开头初始化的动作,如果为了保证其,仅执行一次,也可以放到这个dispatch_once来执行。
然后我们看到它需要一个断言来确定这个代码块是否执行,这个断言的指针要保存起来,相对于第一种方法而言,还需要多保存一个指针。
方法简介中就说的很清楚了:对于在应用中创建一个初始化一个全局的数据对象(单例模式),这个函数很有用。
如果同时在多线程中调用它,这个函数将等待同步等待,直至该block调用结束。
这个断言的指针必须要全局化的保存,或者放在静态区内。使用存放在自动分配区域或者动态区域的断言,dispatch_once执行的结果是不可预知的。
总结:1.这个方法可以在创建单例或者某些初始化动作时使用,以保证其唯一性。2.该方法是线程安全的,所以请放心大胆的在子线程中使用。(前提是你的dispatch_once_t *predicate对象必须是全局或者静态对象。这一点很重要,如果不能保证这一点,也就不能保证该方法只会被执行一次。)
原文地址:http://blog.csdn.net/sammyieveo/article/details/41513465