iOS开发-单例模式

摘要:
什么是Singleton模式?
  • 什么是单例模式?
    >是开发设计模式(共23种)中的1种
    >它可以保证在程序运行过程,一个类只有一个实例(一个对象),而且该实例易于供外界访问,从而方便地控制了实例个数,并节约系统资源
    >使用场合:
    在整个应用程序中,共享一份资源(这份资源只需要创建初始化1次)

  • iOS系统中的一些单例
    UIApplication(应用程序实例) 
    NSNotificationCenter(消息中心)  
    NSFileManager(文件管理)    
    NSUserDefaults(应用程序设置)     
    NSURLCache(请求缓存)   
    NSHTTPCookieStorage(应用程序cookies池)

  • 单例模式的基本实现
    1.懒汉模式
    // 1.在该类中定义一个静态的全局变量,防止被外部用extren访问
    static id _instance;

    /**

     static : 修饰变量

     1> 修饰全局变量

     * 全局变量的作用域仅限于当前文件内部,其他文件不能用extren关键字访问

     2> 修饰局部变量 : 

     * 局部变量的生命周期 跟 全局变量 类似

     * 但是不能改变作用域

     * 能保证局部变量永远只初始化1次,在程序运行过程中,永远只有1分内存

     */

     
    /**
     * 2.重写它这个类的llocWithZone:方法,确保只为你这个类分配一次内存地址
     *  alloc方法内部会调用这个方法
     */
    + (id)allocWithZone:(struct _NSZone *)zone
    {
        if (_instance == nil) { // 防止频繁加锁
            @synchronized(self) { // 加锁,避免在多线程中出错而创建多个对象
                if (_instance == nil) { // 防止创建多次
                    _instance = [super allocWithZone:zone]; 
                }
            }
        }
        return _instance;
    }
    
    // 3.提供一个shared方法让外界调用这个单例(一般单例都会提供这个方法),确保只init一次
    + (instancetype)sharedMusicTool
    {
        if (_instance == nil) { // 防止频繁加锁
            @synchronized(self) {
                if (_instance == nil) { // 防止创建多次
                    _instance = [[self alloc] init];
                }
            }
        }
        return _instance;
    }
    
    // 4.重写copyWithZone:方法,避免使用copy时创建多个对象
    - (id)copyWithZone:(NSZone *)zone
    {
        return _instance;
    }

    2.饿汉模式(当类加载到OC运行环境中(内存)时,就会调用一次生成这个单例(一个类只会加载一次))

    // 1.在该类中定义一个静态的全局变量,防止被外部用extren访问
    static id _instance;
    
    /**
     * 2.重写它这个类的llocWithZone:方法,这里不用加锁,因为程序刚启动,线程还没加载,不会出现线程不安全的问题
     */
    + (id)allocWithZone:(struct _NSZone *)zone
    {
                if (_instance == nil) { // 防止创建多次
                    _instance = [super allocWithZone:zone];
                }
        return _instance;
    }
    
    // 3.提供一个shared方法让外界调用这个单例(一般单例都会提供这个方法)
    + (instancetype)sharedMusicTool
    {
        return _instance;
    }
    
    // 4.重写copyWithZone:方法,避免使用copy时创建多个对象
    - (id)copyWithZone:(NSZone *)zone
    {
        return _instance;
    }
    
    // 5.重写load这个类方法,在里面alloc它
    // 这个方法在程序启动,加载类的时候会调用一次
    + (void)load
    {
      _instance = [[self alloc] init];      
    }
  • 用GCD实现单例模式
    // 1.
    static id _instance;
    
    /**
     * 2.用GCD的dispatch_once方法重写
     */
    + (id)allocWithZone:(struct _NSZone *)zone
    {
        static dispatch_once_t onceToken;
        dispatch_once(&onceToken, ^{
             _instance = [super allocWithZone:zone];
        });
       return _instance;
    }
    
    // 3.
    + (instancetype)sharedMusicTool
    {
        static dispatch_once_t onceToken;
        dispatch_once(&onceToken, ^{
             _instance = [[self alloc] init];
        });
        return _instance;
    }
    
    // 4.
    - (id)copyWithZone:(NSZone *)zone
    {
        return _instance;
    }
  • 非ARC下的单例模式
    // 前面与ACR一样,接着重写以下方法
    // 不做release
    - (oneway void)release {
    }
    
    // retain之后还是自己一份
    - (id)retain {
        return self;
    }
    
    // 计数器永远为1
    - (NSUInteger)retainCount {
        return 1;
    }
    
    // 防止被放进自动计数池释放
    - (id)autorelease {
        return self;
    }
  • 用宏定义实现单例(适配ARC和MRC)
    // 新建一个.h文件或pch文件,把一下代码copy进去。在需要实现单例的类中import进去并使用宏(括号里面传相应的类名)即可
    // 在.h文件使用的宏定义
    #define HJSingletonH(name) + (instancetype)shared##name;
    
    #if __has_feature(objc_arc)
    // 当前的编译器环境是ARC
    // 在.m文件使用的宏定义
    #define HJSingletonM(name) 
    static id _instace; 
    
    + (id)allocWithZone:(struct _NSZone *)zone 
    { 
    static dispatch_once_t onceToken; 
    dispatch_once(&onceToken, ^{ 
    _instace = [super allocWithZone:zone]; 
    }); 
    return _instace; 
    } 
    
    + (instancetype)shared##name 
    { 
    static dispatch_once_t onceToken; 
    dispatch_once(&onceToken, ^{ 
    _instace = [[self alloc] init]; 
    }); 
    return _instace; 
    } 
    
    - (id)copyWithZone:(NSZone *)zone 
    { 
    return _instace; 
    }
    
    #else
    // 当前的编译器环境是MRC
    // 在.m文件使用的宏定义
    #define HJSingletonM(name) 
    static id _instace; 
     
    + (id)allocWithZone:(struct _NSZone *)zone 
    { 
        static dispatch_once_t onceToken; 
        dispatch_once(&onceToken, ^{ 
            _instace = [super allocWithZone:zone]; 
        }); 
        return _instace; 
    } 
     
    + (instancetype)shared##name 
    { 
        static dispatch_once_t onceToken; 
        dispatch_once(&onceToken, ^{ 
            _instace = [[self alloc] init]; 
        }); 
        return _instace; 
    } 
     
    - (id)copyWithZone:(NSZone *)zone 
    { 
        return _instace; 
    } 
    - (oneway void)release {} 
    - (id)retain {return self;} 
    - (NSUInteger)retainCount {return 1;} 
    - (id)autorelease {return self;}
    #endif

免责声明:文章转载自《iOS开发-单例模式》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇NTP 协议介绍Perl中的默认变量$_ 和 @_下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

全面解读:微信服务号升级和群发增至4条的应用方法

全面解读:微信服务号升级和群发增至4条的应用方法 摘要:微信公众平台正式公布微信服务号升级的消息,提供高级群发接口,火速华磊初步总结:群发接口的提供意味着企业获得了精准推送的能力、企业获得了每月多次激活粉丝的能力、企业公众号的粉丝活跃度全面提升、公众号的营销核心过程趋于完美。 2014/4/15 21:58分,微信公众平台正式公布微信服务号升级的消息,消...

SendMessageTimeout 的使用

在WINDOW编程中,发送消息的常用API有SendMessage,PostMessage,PostThreadMessage。 一般每个线程有两个队列:一个用来接收通过Send函数的消息,另外一个队列接收通过Post函数的消息。该两个函数的基本区别是:一个函数需要等待返回的,相当于函数调用,这个是SendMessage;另外一个是将消息放到对方的队列中,...

微信支付商户申请接入信息汇总【接入教程】

微信支付(商户功能)功能介绍 微信支付(商户功能),是公众平台向有出售物品需求的公众号提供推广销售、支付收款、经营分析的整套解决方案。 商户通过自定义菜单、关键字回复等方式向订阅用户推送商品消息,用户可在微信公众号中完成选购支付的流程。商户也可以把商品网页生成二维码,张贴在线下的场景,如车站和广告海报。用户扫描后可打开商品详情,在微信中直接购买。 场景及类...

Linux 长时间操作设置不断开

1、第一次尝试失败 修改/etc/ssh/sshd_config文件, 找到 ClientAliveInterval 0 ClientAliveCountMax 3 并将注释符号("#")去掉, 将ClientAliveInterval对应的0改成60, ClientAliveInterval指定了服务器端向客户端请求消息 的时间间隔, 默认是0, 不发送...

CRLF注入原理

CRLF 指的是回车符(CR,ASCII 13, ,%0d) 和换行符(LF,ASCII 10, ,%0a),操作系统就是根据这个标识来进行换行的,你在键盘输入回车键就是输出这个字符,只不过win和linux系统采用的标识不一样而已。 在HTTP当中HTTP的Header和Body之间就是用两个crlf进行分隔的,如果能控制HTTP消息头中的字符,注入一些...

内核打印等级

内核打印日志等级配置存放在/proc/sys/kernel/printk,默认6   4   1   7 上面显示的4个数据分别对应: 控制台日志级别:优先级高于该值的消息将被打印至控制台 默认的消息日志级别:将用该优先级来打印没有优先级的消息 最低的控制台日志级别:控制台日志级别可被设置的最小值(最高优先级) 默认的控制台日志级别:控制台日志级别的缺省...