所有转出博客园,请您注明出处: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单例的两种安全实现方案的更多相关文章

  1. iOS单例的两种实现

    单例模式算是开发中比较常见的一种模式了.在iOS中,单例有两种实现方式(至少我目前只发现两种).根据线程安全的实现来区分,一种是使用@synchronized,另一种是使用GCD的dispatch_o ...

  2. iOS 创建单例的两种方法

    创建一个单例很多办法.我先列举一个苹果官方文档中的写法. [cpp] view plaincopy static AccountManager *DefaultManager = nil; + (Ac ...

  3. 【iOS开发】创建单例的两种方法

    创建一个单例很多办法.我先列举一个苹果官方文档中的写法. [cpp] view plaincopy   static AccountManager *DefaultManager = nil; + ( ...

  4. IOS创建单例的两种方法

    1.0  苹果官方写法:  static AccountManager *DefaultManager = nil; + (AccountManager *)defaultManager { if ( ...

  5. java单例的几种实现方法

    java单例的几种实现方法: 方式1: public class Something { private Something() {} private static class LazyHolder ...

  6. 属性传值,协议传值,block传值,单例传值四种界面传值方式

    一.属性传值 对于属性传值而言,相对于其它的三种 方法来说,是最基础,最简单的一种 方法,但,属性传值 有很大的局限性,因为是适用于第一个界面向第二个界面传 值,第二个向第三个界面传值等等.N界面向N ...

  7. Objective-C和Swift实现单例的几种方式

    在Swift开发中,我们对于跨类调用的变量常量,因为并没有OC中使用的全局头文件中写宏的形式,我们一般采用在类外定义全局变量/常量的形式来跨类调用.而问题在于目前写的项目需要在新添加的OC写的功能模块 ...

  8. 「Android」单例的五种写法

    单例 发现博客园可以很好的设置自己的博客文章的展示,很开心,然后特此发一篇 其实这几种写法大家应该都会的,就权当拿来记录一下吧,以后复习巩固也比较方便. 这篇文章中的代码,来自一篇视频(我想找视频贴上 ...

  9. Java并发编程中的设计模式解析(二)一个单例的七种写法

    Java单例模式是最常见的设计模式之一,广泛应用于各种框架.中间件和应用开发中.单例模式实现起来比较简单,基本是每个Java工程师都能信手拈来的,本文将结合多线程.类的加载等知识,系统地介绍一下单例模 ...

随机推荐

  1. Java面试通关要点【问题汇总篇】

    基础篇 基本功: 面向对象的特征 final, finally, finalize 的区别 int 和 Integer 有什么区别 重载和重写的区别 抽象类和接口有什么区别 说说反射的用途及实现 说说 ...

  2. Android实现图片下载并保存SD卡

    一.首先获取图片 //第一种获取图片的方法 String filePath = downloadUrl; //以下是取得图片的方法 取得的是InputStream,直接从InputStream生成bi ...

  3. jquery 之事件 方法

    一.jquery事件 1 blur()       触发.或将函数绑定到指定元素的 blur 事件 2 change()     触发.或将函数绑定到指定元素的 change 事件 3 click() ...

  4. .net core Web应用启动类

    在ASP.NET Core中,Startup类为Web应用的入口类,用于配置Web服务的管道/过滤器以及Web应用所能用到的服务.在启动Web应用后,ASP.NET将在主库中查询名为Startup的类 ...

  5. java将int类型的变量转化成String类型的

    第一种方法:String的valueOf方法,int i=5;String s=String.valueOf(i);第二种方法,直接在int后面加一个空的字符串,因为在java里面,默认任务int类型 ...

  6. HTML5-入门。

    什么是HTML5? HTML5是超文本语言,不是编程语言,html5是html语言的最新版本,需要注意浏览器的兼容性问题. HTML5技术一般是指的是HTML5.CSS3.JavaScript三种技术 ...

  7. Windows Azure系列公开课 - 第三课:创建虚拟机 (基础篇)

    Windows Azure微软智能云平台主要提供四大类服务:计算服务(Compute),数据服务 (Data Services) ,应用服务 (App Services) ,网络服务(Network) ...

  8. Ionic开发项目

    hybrid app是移动开发代替原生开发完成app应用项目的一种方案,Ionic是hybrid app开发的一种选择.对ionic有兴趣可以去网上找相应的基础知识来学习了解,因为Ionic是基于An ...

  9. 基于双下划线的跨表查询 (join查询)

    因为你的数据库中的查询就是重点  那么你的django提供的orm也是查询语句最重点 ,也提供的查询方法比较的多,下面我们学习下类似于MYSQL的连表(join)查询 Django 还提供了一种直观而 ...

  10. 五、python小功能记录——打包程序

    使用pyinstaller打包Python程序 安装工具 :pip3 install pyinstaller 在Python程序文件夹上(不点进去)按住shift并且右键,在弹出的选项中点击" ...