objc单例的两种安全实现方案
所有转出博客园,请您注明出处:http://www.cnblogs.com/xiaobajiu/p/4122034.html
objc的单例的两种安全实现方案
首先应该知道单例的实现有两大类,一个是懒汉式,一个是饿汉式。所谓的懒汉式就是在我们用到某对象(资源)的时候,去问一个负责提供的方法要那个对象。那个方法发现没有这个资源时就去创建资源,如果是已经有该资源时就直接返回这个资源。而饿汉式就是那个负责提供的方法早已为我们准备好了我们想要的资源问它,它就提供给我们那个它早就准备好了的资源。
饿汉式的实现是简单而且好理解,但是它的理念不适合移动设备,因为饿汉式提前占用了内存,却不管我们需不需要。所以饿汉式的实现在最后作为一点补充。我们主要使用懒汉式,它符合移动的开发理念。
首先理清实现单例模式(懒汉式)的思路:
1.依靠什么来单例?
答:依靠静态变量,因为它在内存中只有一份。只要判断这个变量存不存在我们就知道需不需要创建一个对象赋值给那个静态变量。
2.但是在并发的情况下,我们同时去判断一个变量都得到了nil的结果,这两者就会都去创建一个新对象,那么会导致唯一性失效,就会使这两次访问得到不同的结果了,怎么解决?
答:加锁或者使用dispatch_once。
请先看加锁的方案。多说两句,加锁@synchronized()的效率可不是java,c#那么高的。《cocao开发者编程手册》47页这样写道:“这需要精确的垃圾回收,以及重写本地二进制文件的能力,Objectic-C编译器这两件事都做不了。所以,这个关键字非常低效”。但其实我们是可以避免频繁加锁,在网上找到的大多数代码都没有考虑到频繁加锁的问题的,而这里解决了这个问题,直接看代码吧:
//
// God.h
#import <Foundation/Foundation.h> @interface God : NSObject<NSCopying>
+ (God*)sharedGod; @end
//
// God.m #import "God.h"
static God *_singleGod; @implementation God
+ (id)allocWithZone:(struct _NSZone *)zone
{
if(!_singleGod){//防止频繁加锁
@synchronized([God class]){
if(!_singleGod)
_singleGod= [super allocWithZone:zone];
}
}
return _singleGod;
}
+ (God*)sharedGod
{
if(!_singleGod){//防止频繁加锁
@synchronized([God class]){
if(!_singleGod){
_singleGod= [[self alloc] init];
}
}
}
return _singleGod;
}
- (id)copyWithZone:(NSZone *)zone
{
return _singleGod;
}
//MRC下增加下面四个方法即可
- (id)retain
{
return self;
}
- (NSUInteger)retainCount
{
return ;
}
- (oneway void)release
{}
- (id)autorelease
{
return self;
}
@end
加锁的方案看起来似乎过于臃肿,使用GCD提供的dispatch_once来解决,将会使代码看起来简洁有力。(注:dispatch_once是GCD提供的一次性代码方案,该方法在整个程序的生命周期中只会执行一次。下一次执行到这里不会再执行了。)
dispatch_once版的实现如下:
//
// God2.m #import "God2.h"
static God2 *singleGod2;
@implementation God2
+ (id)allocWithZone:(struct _NSZone *)zone
{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
singleGod2= [super allocWithZone:zone];
});
return singleGod2;
}
+ (instancetype)sharedGod2
{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
singleGod2= [[self alloc] init];
});
return singleGod2;
} - (id)copyWithZone:(NSZone *)zone
{
return singleGod2;
} //MRC下增加下面四个方法即可
- (id)retain
{
return self;
}
- (NSUInteger)retainCount
{
return ;
}
- (oneway void)release
{
}
- (id)autorelease
{
return self;
}
@end
使用dispatch_once来实现单例,我们更本不需要担心线程安全问题,因为GCD提供的解决方案本身就是线程安全的。大大简化了开发难度。
最后简单的补充一点关于懒汉式的写法,那么要知道+(void)load这个方法。它是类型加载到runtime时调用该方法,并且仅此一次。它的核心就是下面一行代码:
+ (void)load
{
singleGod3= [[self alloc] init];
}
欢迎各位朋友指正,以免误人子弟。
objc单例的两种安全实现方案的更多相关文章
- iOS单例的两种实现
单例模式算是开发中比较常见的一种模式了.在iOS中,单例有两种实现方式(至少我目前只发现两种).根据线程安全的实现来区分,一种是使用@synchronized,另一种是使用GCD的dispatch_o ...
- iOS 创建单例的两种方法
创建一个单例很多办法.我先列举一个苹果官方文档中的写法. [cpp] view plaincopy static AccountManager *DefaultManager = nil; + (Ac ...
- 【iOS开发】创建单例的两种方法
创建一个单例很多办法.我先列举一个苹果官方文档中的写法. [cpp] view plaincopy static AccountManager *DefaultManager = nil; + ( ...
- IOS创建单例的两种方法
1.0 苹果官方写法: static AccountManager *DefaultManager = nil; + (AccountManager *)defaultManager { if ( ...
- java单例的几种实现方法
java单例的几种实现方法: 方式1: public class Something { private Something() {} private static class LazyHolder ...
- 属性传值,协议传值,block传值,单例传值四种界面传值方式
一.属性传值 对于属性传值而言,相对于其它的三种 方法来说,是最基础,最简单的一种 方法,但,属性传值 有很大的局限性,因为是适用于第一个界面向第二个界面传 值,第二个向第三个界面传值等等.N界面向N ...
- Objective-C和Swift实现单例的几种方式
在Swift开发中,我们对于跨类调用的变量常量,因为并没有OC中使用的全局头文件中写宏的形式,我们一般采用在类外定义全局变量/常量的形式来跨类调用.而问题在于目前写的项目需要在新添加的OC写的功能模块 ...
- 「Android」单例的五种写法
单例 发现博客园可以很好的设置自己的博客文章的展示,很开心,然后特此发一篇 其实这几种写法大家应该都会的,就权当拿来记录一下吧,以后复习巩固也比较方便. 这篇文章中的代码,来自一篇视频(我想找视频贴上 ...
- Java并发编程中的设计模式解析(二)一个单例的七种写法
Java单例模式是最常见的设计模式之一,广泛应用于各种框架.中间件和应用开发中.单例模式实现起来比较简单,基本是每个Java工程师都能信手拈来的,本文将结合多线程.类的加载等知识,系统地介绍一下单例模 ...
随机推荐
- ArcGIS Pro 自定义坐标系地图矢量切片制作
ArcGIS Pro从1.4版本起就支持自定义坐标系统地图的矢量切片制作了. 步骤: 1. 将地图有全图范围缩小到屏幕像素大约10*10像素的范围,然后记录下地图的比例尺.这一步十分关键,不然系统要经 ...
- How to save rules of the iptables?
The easy way is to use iptables-persistent. Install iptables-persistent: sudo apt-get install iptabl ...
- Hadoop & Spark & Hive & HBase
Hadoop: http://hadoop.apache.org/docs/r2.6.4/hadoop-project-dist/hadoop-common/SingleCluster.html bi ...
- Python爬虫教程-02-使用urlopen
Spider-02-使用urlopen 做一个最简单的python爬虫,使用爬虫爬取:智联招聘某招聘信息的DOM urllib 包含模块 - urllib.request:打开和读取urls - ur ...
- 字典构造、合并(dict)、排序
使用dict,zip方法将两个list合并为dict keys = ["b", "a", "c", "e", " ...
- Exchange Server 2007 多名称证书配置
Exchange Server 2007上配置多名称证书,有两种方式,一种是通过Exchange Management Shell利用命令行工具进行创建:另一种是通过证书管理器控制台进行创建.本文将介 ...
- 使用ESP8266连接到Azure
很多同学在问我,ESP8266如何连接到Azure,因为官方只有Spark Fun之类的DevKit(开发套件)出了教程(我只找到了套件的文档). 先上视频:http://v.youku.com/v_ ...
- 【Azure IoT DevKit】实验终于做完了
大家好,我是MSP李桑榆 今天终于把几个Azure IoT DevKit的小实验的视频给做完了. 不敢说什么指导,只是给大家一个参考.因为Devkit不需要你写一行代码,只需要你按着步骤来,并没有什么 ...
- 021.4 IO流——字节、字符桥梁(编码解码)
默认使用的就是gbk编码,这里的例子改成了utf8编码 写入—编码 private static void writeText() throws IOException { FileOutputStr ...
- [零基础学JAVA]Java SE面向对象部分.面向对象基础(03)
1.静态变量的使用 2.单例模式的使用. 3.构造方法的私有化. 4.string的使用,两种构造的不同. 小的记忆错误: · 数组的长度:数组名称.length 这个没()哈~~ · 字符 ...