zoukankan      html  css  js  c++  java
  • 【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
    //============================================================================


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





  • 相关阅读:
    MongoDB ‘conn’Mongo 对象远程代码执行漏洞
    Linux Kernel 本地拒绝服务漏洞
    Linux Kernel ‘skbuff.c’本地拒绝服务漏洞
    WordPress Citizen Space插件跨站请求伪造漏洞
    OpenSSH远程拒绝服务漏洞
    Bug之王花落谁家:四大最危险编程语言,PHP竟然不是bug最多的语言!
    《出Bug表》假如诸葛亮是程序员!写Bug测Bug,不宜异同!
    关于程序员的段子,有没有get到你的点?单身的程序员才是完整的程序员!
    关于程序员的段子,有没有get到你的点?单身的程序员才是完整的程序员!
    socket 通信 入门3 android 客户端 C# 服务端
  • 原文地址:https://www.cnblogs.com/gcczhongduan/p/6684487.html
Copyright © 2011-2022 走看看