单例的应用十分普遍,单例模式使一个类仅仅有一个实例

*易于供外界訪问.
*方便控制实例个数,节约系统资源.

*OC中的常见单例:
如:UIApplication,  NSNotificationCenter,  NSUserDefaults, NSFIleManager。

*应用程序中用到的单例:
如:背景音乐。音效管理等。

一、ARC中实现单例

创建单例的步骤:
*1.定义一个全局的静态变量_instance,用来记录“第一次”被实例化出来的对象.
*2.重写allocWithZone方法,此方法是为对象分配内存空间必须会被调用的一个方法!
因此。在此方法中使用“dispatch_once”,可以保证在多线程中,_instance也仅仅能被“分配”一次空间.
*3.定义一个sharedXXX“类”方法。方便其它使用单例的对象调用此单例.
在此方法中,相同使用“dispatch_once”,保证使用类方法调用的对象,仅仅会被初始化一次。
凝视:假设不考虑copy& MRC,以上三个步骤就可以!
*4.假设要支持copy,则须要:
(1)遵守NSCopying协议
(2)在copyWithZone方法中,直接返回_instance

tips:

*一般的写法(懒汉式, 饿汉式,
加锁):
if(!_instance)_instance=[[XNShareToolalloc]init];
return_instance;
*懒汉式是线程不安全的.因此实际中不这么写. 还有饿汉式,加锁等.

*可是OC中有其自己的写法.须要结合其对象生命周期的一些方法来写单例.

*为什么要使用dispatch_one?

:

防止多线程同一时候进来,就相当与Java单例中的加锁机制,保证仅仅被实例化一次.
但这里使用的不是synchronized, 是类似相互排斥锁的东西, 但比他的性能高.

ARC中实现单例的代码例如以下:
  1. @implementation XNShareTool
  2. /**
  3. 步骤:
  4. 1.一个静态变量_inastance
  5. 2.重写allocWithZone, 在里面用dispatch_once, 并调用super allocWithZone
  6. 3.自己定义一个sharedXX, 用来获取单例. 在里面也调用dispatch_once, 实例化_instance
  7. -----------可选------------
  8. 4.假设要支持copy. 则(先遵守NSCopying协议)重写copyWithZone, 直接返回_instance就可以.
  9.  
  10. */
  11. /**第1步: 存储唯一实例*/
  12. static XNShareTool *_instance;
  13.  
  14. /**第2步: 分配内存孔家时都会调用这种方法. 保证分配内存alloc时都同样*/
  15. +(id)allocWithZone:(struct _NSZone *)zone{
  16. //调用dispatch_once保证在多线程中也仅仅被实例化一次
  17. static dispatch_once_t onceToken;
  18. dispatch_once(&onceToken, ^{
  19. _instance = [super allocWithZone:zone];
  20. });
  21. return _instance;
  22. }
  23.  
  24. /**第3步: 保证init初始化时都同样*/
  25. +(instancetype)sharedTool{
  26. static dispatch_once_t onceToken;
  27. dispatch_once(&onceToken, ^{
  28. _instance = [[XNShareTool alloc] init];
  29. });
  30. return _instance;
  31. }
  32.  
  33. /**第4步: 保证copy时都同样*/
  34. -(id)copyWithZone:(NSZone *)zone{
  35. return _instance;
  36. }
  37.  
  38. @end

測试代码例如以下(打印单例对象的地址都同样):

  1. -(void)viewDidLoad{
  2. //实例化一个类的几种方法. 单例就是要保证实例化出来的类是同一个类
  3. //1.alloc init方法. 一般不这么来调用单例.
  4. XNShareTool *t1 = [[XNShareTool alloc] init];
  5. XNShareTool *t2 = [[XNShareTool alloc] init];
  6.  
  7. //2.类方法
  8. XNShareTool *t3 = [XNShareTool sharedTool];
  9.  
  10. //3.copy
  11. XNShareTool *t4 = [t3 copy];
  12.  
  13. NSLog(@"%@ %@ %@ %@", t1, t2, t3, t4);
  14. }

二、MRC中运用单例

由于单例对象是用static标记过的, 因此存放在静态区.
所以在MRC中不须要由程序猿去管理,因此要去覆盖一些内存管理的方法.

实现部分与ARC一致,仅仅须要覆盖一些MRC中内存管理的方法:
*- (id)retain.  单例中不须要添加引用计数器.returnself.
*- (id)autorelease.  仅仅有堆中的对象才须要.单例中不须要.returnself.
*- (NSUInteger)retainCount.(可写可不写,防止引起误解).单例中不须要改动引用计数。返回最大的无符号整数就可以.return
UINT_MAX;
*- (oneway void)release.不须要release.直接覆盖,生命也不做.
  1. #import "XNShareTool.h"
  2.  
  3. @implementation XNShareTool
  4.  
  5. static XNShareTool *_instance;
  6.  
  7. + (id)allocWithZone:(struct _NSZone *)zone {
  8. static dispatch_once_t onceToken;
  9. dispatch_once(&onceToken, ^{
  10. _instance = [super allocWithZone:zone];
  11. });
  12. return _instance;
  13. }
  14.  
  15. + (instancetype)sharedTool {
  16. static dispatch_once_t onceToken;
  17. dispatch_once(&onceToken, ^{
  18. _instance = [[XNShareTool alloc] init];
  19. });
  20. return _instance;
  21. }
  22.  
  23. - (id)copyWithZone:(NSZone *)zone {
  24. return _instance;
  25. }
  26.  
  27. #pragma mark - MRC中须要覆盖的方法
  28. //不须要计数器+1
  29. - (id)retain {
  30. return self;
  31. }
  32.  
  33. //不须要. 堆区的对象才须要
  34. - (id)autorelease {
  35. return self;
  36. }
  37.  
  38. //不须要
  39. - (oneway void)release {
  40. }
  41.  
  42. //不须要计数器个数. 直接返回最大无符号整数
  43. - (NSUInteger)retainCount {
  44. return UINT_MAX; //參照常量区字符串的retainCount
  45. }
  46.  
  47. @end

三、ARC与MRC的整合


整合是为了方便单例既能在ARC中使用,又能在MRC中使用。

而不必去改动单例中的方法。


详细做法是使用宏定义:(推断是否是ARC环境,是的话就省略内存管理的方法)

#if !__has_feature(objc_arc)

MRC中内存管理的方法放在这个地方

#endif

代码例如以下:

  1. //=============================ARC/MRC整合=======================================
  2. #pragma mark - MRC中须要覆盖的方法, ARCMRC的整合
  3. #if !__has_feature(objc_arc)
  4. - (id)retain {
  5. return self;
  6. }
  7.  
  8. - (id)autorelease {
  9. return self;
  10. }
  11.  
  12. - (oneway void)release {
  13. }
  14.  
  15. - (NSUInteger)retainCount {
  16. return UINT_MAX;
  17. }
  18.  
  19. #endif
  20. //============================================================================

转载请注明出处:http://blog.csdn.net/xn4545945

【iOS】ARC-MRC下的单例及其应用的更多相关文章

  1. iOS: ARC & MRC下string内存管理策略探究

    ARC & MRC下string内存管理策略探究 前两天跟同事争论一个关于NSString执行copy操作以后是否会发生变化,两个人整了半天,最后写代码验证了一下,发现原来NSString操作 ...

  2. ARC模式下的单例写法

    // 单例 + (id)sharedInstance { __strong static id sharedObject = nil; static dispatch_once_t onceToken ...

  3. iOS ARC环境下dealloc的使用

    众所周知,iOS开发的时候,使用ARC的话,dealloc函数是不需要实现的,写了反而会出错. 但有些特殊的情况,dealloc函数还是需要的. 比如,在画面关闭的时候,需要把ViewControll ...

  4. iOS 页面间传值 之 单例传值 , block 传值

    ios 页面间传值有许多,前边已经分享过属性传值和代理传值,今天主要说一下单例传值和 block 传值 单例传值:单例模式一种常用的开发的模式,单例因为在整个程序中无论在何时初始化对象,获取到的都是同 ...

  5. iOS中浅淡UIApplication单例-b

    在iOS的操作系统中 每一个程序都对应一个application单例,每一个application都对应一个Appdelegate代理,在代理中控制程序的各个状态.我们在程序中获取Applicatio ...

  6. IOS设计模式第二篇之单例设计模式

    现在我们的组件已经有组织了.你需要从其他的地方得到数据,你也可以创建一个API类管理数据这个下个设计模式单例里面介绍. 这个单例设计模式确保这个类仅仅拥有一个实例,并且为这个实例提供一个全局的访问点. ...

  7. iOS开发中常用的单例

    定义:一个类的对象,无论在何时创建.无论创建多少次,创建出来的对象都是同一个对象. 使用场景:当有一些数据需要共享给别的类的时候,就可以把这些数据保存在单例对象中.   关键代码: + (instan ...

  8. iOS——Swift开发中的单例设计模式(摘译,非原创)

    最近在开发一个小的应用,遇到了一些Objective-c上面常用的单例模式,但是swift上面还是有一定区别的,反复倒来倒去发现不能按常理(正常的oc to swift的方式)出牌,因此搜索了一些帖子 ...

  9. 多线程下的单例-double check

    话不多说直接上代码: public sealed class Singleton { private static Singleton _instance = null; // Creates an ...

随机推荐

  1. 怎样将一个Long类型的数据转换成字节数组

    直接上代码: //先写进去 long n = 1000000L; ByteArrayOutputStream baos = new ByteArrayOutputStream(); DataOutpu ...

  2. poj 1386 Play on Words门上的单词【欧拉回路&&并查集】

    题目链接:http://poj.org/problem?id=1386 题目大意:给你若干个字符串,一个单词的尾部和一个单词的头部相同那么这两个单词就可以相连,判断给出的n个单词是否能够一个接着一个全 ...

  3. SpringBoot邮件发送

    这篇文章介绍springboot的邮件发送. 由于很简单就没有分出server和imp之类,只是在controller简单写个方法进行测试. 首先pom文件加入spring-boot-starter- ...

  4. XamarinSQLite教程在Xamarin.iOS项目中定位数据库文件

    XamarinSQLite教程在Xamarin.iOS项目中定位数据库文件 开发者可以在指定的路径中找到复制的数据库文件,具体的操作步骤如下: (1)单击Mac电脑中Finder菜单中的“前往”|“前 ...

  5. seq2seq升级TF1.5后_Linear报错

    解决TF升级到1.5之后seq2seq.py出现的引用报错: 1.4时候使用rnn_cell_impl的_Linear没有问题的,TF升级到1.5之后这一行就运行不过去了,查到的方法是引用core_r ...

  6. 双向BFS—>NOIP2002 字串变换

    如果目标也已知的话,用双向BFS能很大提高速度 单向时,是 b^len的扩展. 双向的话,2*b^(len/2)  快了很多,特别是分支因子b较大时 至于实现上,网上有些做法是用两个队列,交替节点搜索 ...

  7. Linux之经典互联网架构

    经典互联网架构 netstat -tulnp |grep 80ss -tulnp|grep 80 前期铺垫: 1. Linux要能上网2. 掌握Linux软件包安装方法2.1 rpm包管理 2.1.1 ...

  8. Docker安装管理界面portainer

    在Ubuntu或者Debian已经部署完毕Docker 拉取镜像文件: sudo docker pull docker.io/portainer/portainer Using default tag ...

  9. Cow Exhibition [POJ2184] [DP] [背包的负数处理]

    题意: 有很多羊,每只羊有一个幽默度和智商,要选出一些羊,智商加幽默度总和最大,其中智商总和和幽默度总和都不能是负数. 样例输入: 5 -5 7 8 -6 6 -3 2 1 -8 -5 样例输出: 8 ...

  10. .net 4.0 中的特性总结(四):Tuple类型

    Tuple是具有指定数量和顺序的值的一种数据结构.针对这种数据结构,.Net4.0中提供了一组Tuple类型,具体如下: Tuple   Tuple<T>   Tuple<T1, T ...