【iOS】ARC-MRC下的单例及其应用

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

*易于供外界访问.

*方便控制实例个数,节约系统资源.

*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中实现单例的代码如下:

@implementation XNShareTool
/**
 步骤:
 1.一个静态变量_inastance
 2.重写allocWithZone, 在里面用dispatch_once, 并调用super allocWithZone
 3.自定义一个sharedXX, 用来获取单例. 在里面也调用dispatch_once, 实例化_instance
 -----------可选------------
 4.如果要支持copy. 则(先遵守NSCopying协议)重写copyWithZone, 直接返回_instance即可.

 */
/**第1步: 存储唯一实例*/
static XNShareTool *_instance;

/**第2步: 分配内存孔家时都会调用这个方法. 保证分配内存alloc时都相同*/
+(id)allocWithZone:(struct _NSZone *)zone{
    //调用dispatch_once保证在多线程中也只被实例化一次
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        _instance = [super allocWithZone:zone];
    });
    return _instance;
}

/**第3步: 保证init初始化时都相同*/
+(instancetype)sharedTool{
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        _instance = [[XNShareTool alloc] init];
    });
    return _instance;
}

/**第4步: 保证copy时都相同*/
-(id)copyWithZone:(NSZone *)zone{
    return _instance;
}

@end

测试代码如下(打印单例对象的地址都相同):

-(void)viewDidLoad{
    //实例化一个类的几种方法. 单例就是要保证实例化出来的类是同一个类
    //1.alloc init方法. 一般不这么来调用单例.
    XNShareTool *t1 = [[XNShareTool alloc] init];
    XNShareTool *t2 = [[XNShareTool alloc] init];

    //2.类方法
    XNShareTool *t3 = [XNShareTool sharedTool];

    //3.copy
    XNShareTool *t4 = [t3 copy];

    NSLog(@"%@ %@ %@ %@", t1, t2, t3, t4);
}

二、MRC中运用单例

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

实现部分与ARC一致,只需要覆盖一些MRC中内存管理的方法:

*- (id)retain.  单例中不需要增加引用计数器.returnself.

*- (id)autorelease.  只有堆中的对象才需要.单例中不需要.returnself.

*- (NSUInteger)retainCount.(可写可不写,防止引起误解).单例中不需要修改引用计数,返回最大的无符号整数即可.return UINT_MAX;

*- (oneway void)release.不需要release.直接覆盖,生命也不做.

#import "XNShareTool.h"

@implementation XNShareTool

static XNShareTool *_instance;

+ (id)allocWithZone:(struct _NSZone *)zone {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        _instance = [super allocWithZone:zone];
    });
    return _instance;
}

+ (instancetype)sharedTool {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        _instance = [[XNShareTool alloc] init];
    });
    return _instance;
}

- (id)copyWithZone:(NSZone *)zone {
    return _instance;
}

#pragma mark - MRC中需要覆盖的方法
//不需要计数器+1
- (id)retain {
    return self;
}

//不需要. 堆区的对象才需要
- (id)autorelease {
    return self;
}

//不需要
- (oneway void)release {
}

//不需要计数器个数. 直接返回最大无符号整数
- (NSUInteger)retainCount {
    return UINT_MAX;  //参照常量区字符串的retainCount
}

@end

三、ARC与MRC的整合

整合是为了方便单例既能在ARC中使用,又能在MRC中使用。而不必去修改单例中的方法。

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

#if !__has_feature(objc_arc)

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

#endif

代码如下:

//=============================ARC/MRC整合=======================================
#pragma mark - MRC中需要覆盖的方法, ARC与MRC的整合
#if !__has_feature(objc_arc)
- (id)retain {
    return self;
}

- (id)autorelease {
    return self;
}

- (oneway void)release {
}

- (NSUInteger)retainCount {
    return UINT_MAX;
}

#endif
//============================================================================
时间: 2024-10-12 21:33:44

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

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

ARC & MRC下string内存管理策略探究 前两天跟同事争论一个关于NSString执行copy操作以后是否会发生变化,两个人整了半天,最后写代码验证了一下,发现原来NSString操作没我们想的那么简单,下面就让我们一起看看NSString和NSMutableString在MRC下执行retain,copy,mutableCopy,以及ARC下不同的修饰__weak, __strong修饰赋值究竟发生了什么. 一.验证代码如下: - (void)testStringAddress { i

黑马程序员——多线程下的单例设计模式的安全问题

//多线程下的单例设计模式 class Sing { //饿汉式不存在安全问题,因为其不是线程同步的 private static Sing s = new Sing(); private Sing(){} public static Sing getInstance() { return s; } } class Single { private static Single s = null; private Single(){} public static Single getInstanc

Swift语言下的单例设计模式实现(SINGLETON)

一.意图 保证一个类公有一个实例,并提供一个访问它的全局访问点. 二.使用场景 1.使用场景 当类只能有一个实例而且客户可以从一个众所周知的访问点访问它时 当这个唯一实例应该是通过子类化可扩展的,并且客户应该无需更改代码就能使用一个扩展的实例时. 2.实现的重要三个步骤 私有化构造方法(Swift不支持) 使用一个静态变量保存实例的引用 提供一个全局的访问方法 三. Swift语言下的实现 Swift语言不支持变量及方法的权限,没有办法隐藏变量及方法,可以随意直接创建一个实例.单例的创建有很多写

C#下的单例实现

本文介绍了C#下关于单例的三种实现方法. 最普通的一种: 以下是引用片段: using System; public class Singleton { private static Singleton instance; private Singleton() {} public static Singleton Instance { get { if (instance == null) instance = new Singleton(); return instance; } } } 这

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

ios 页面间传值有许多,前边已经分享过属性传值和代理传值,今天主要说一下单例传值和 block 传值 单例传值:单例模式一种常用的开发的模式,单例因为在整个程序中无论在何时初始化对象,获取到的都是同一个对象,对象的属性相同,所以可以用来传值. block 传值 与 代理传值相似,主要用于第二个页面向第一个页面传值,block 传值具体步骤: 在第二个页面: 1.声明: block typedef void(^SendMessagerBlock) (NSString *str); 2.创建方法:

iOS开发之自定义一个单例

这里我使用宏: // .h#define single_interface(class)  + (class *)shared##class; // .m// \ 代表下一行也属于宏// ## 是分隔符 #define single_implementation(class) \ static class *_instance; \ \ + (class *)shared##class \ { \ if (_instance == nil) { \ _instance = [[self allo

ARC\MRC下property关键字的用法

使用属性关键字前,需要先理解属性关键字的作用.其实<简单来说...>就是指导底层如何生成set.get方法的,如果不自己定义set.get方法,系统将会自动为你生成,但如何生成依据的就是这些属性关键字,不同的属性关键字的组合将决定set.get方法的具体实现,实现不同,功能自然也就不同了... -----------------------------------------------------------------------------------------------------

iOS ARC环境下dealloc的使用

众所周知,iOS开发的时候,使用ARC的话,dealloc函数是不需要实现的,写了反而会出错. 但有些特殊的情况,dealloc函数还是需要的. 比如,在画面关闭的时候,需要把ViewController的某些资源释放, 在viewDidDissppear不一定合适,viewDidUnload一般情况下只在memory warning的时候才被调用. 不用ARC的情况下,我们自然会想到dealloc函数. 其实ARC环境下,也没有把dealloc函数禁掉,还是可以使用的.只不过不需要调用[sup

iOS开发中常用的单例

定义:一个类的对象,无论在何时创建.无论创建多少次,创建出来的对象都是同一个对象. 使用场景:当有一些数据需要共享给别的类的时候,就可以把这些数据保存在单例对象中. 关键代码: + (instancetype)allocWithZone:(struct_NSZone *)zone {     static id instance = nil;     if(instance == nil)     {       instance =   [super allocWithZone:zone];