ios 单例模式的写法

来源:互联网 发布:高洛峰 php教程 编辑:程序博客网 时间:2024/05/18 02:28



1. 所谓的单例在于“单”,也就是唯一的,每次的调用,使用会只分配一个内存空间给需要的“用户”


//=========================


.h 


//

//  WWCUser.h

//  TestCAOrUIViewAnimationApp7-30

//

//  Created by Whitney.c on 15/8/11.

//  Copyright (c) 2015 ZhongShan Sun union Medical Technology Co. Ltd. All rights reserved.

//


#import <Foundation/Foundation.h>


@interface WWCUser : NSObject



+(instancetype) sharedInstance;

@end



//=========================

 .m 文件


//

//  WWCUser.m

//  TestCAOrUIViewAnimationApp7-30

//

//  Created by Whitney.c on 15/8/11.

//  Copyright (c) 2015 ZhongShan Sun union Medical Technology Co. Ltd. All rights reserved.

//


#import "WWCUser.h"


@implementation WWCUser




// 有时候我或许就这样写了

// 如下:

+(instancetype) sharedInstance

{

    staticWWCUser *_wwcuser  = nil;

    

    staticdispatch_once_t onceToken ;

    

    dispatch_once(&onceToken , ^{

        _wwcuser = [[selfalloc] init];

    });

    

    return _wwcuser;

}


// 这样的写法,然后并不能保证其唯一性

// 因为"用户"可能会 ; WWCUser *wwcuser = [WWCUser alloc] init];

// 也可能会是调用上面的普通单例写法; WWCUser *wwcuser = [WWCuser sharedInstance];


// 然而输出上面2个写法的结果却不是一样的 也证明这种普通的单例写法是不正确的


// 大家知道 alloc是分配内存,init是初始化

// alloc 分配内存时是会调用 allocWithZone方法的

// 官方的文档有这样写 (注意:allocWithZone )


//  static AccountManager *DefaultManager = nil;

//  + (AccountManager *)defaultManager {

//      if (!DefaultManager) DefaultManager = [[self allocWithZone:NULL] init];

//      return DefaultManager;

//  }



// 当有拷贝对象的时候,或许也是同样的原理

// 可以都让它返回,上面的第一种写法的结果( [WWCuser sharedInstance] )



// 下面说下具体的写法吧 ,就是重写 allocWithZone  copyWithZone 写法


+(id)allocWithZone:(struct_NSZone *)zone

{

    return  [WWCUsersharedInstance];

}


+(id)copyWithZone:(struct_NSZone *)zone

{

    return  [WWCUsersharedInstance];

}


// 在输出结果显示,结果是一样的


@end



//  看别人的文章得出 还有另外一种写法 也就是IOS4.0 后的写法

+ (WWCUser *)sharedManager

{

    staticWWCUser *sharedUserInstance = nil;

    staticdispatch_once_t predicate_once;

    dispatch_once(&predicate_once, ^{

        sharedUserInstance = [[selfalloc] init];

    });

    return sharedUserInstance;

}


// 上面的代码主要是 体现出了 dispatch_once 函数的作用


我们看到,该方法的作用就是执行且在整个程序的声明周期中,仅执行一次某一个block对象。简直就是为单例而生的嘛。而且,有些我们需要在程序开头初始化的动作,如果为了保证其,仅执行一次,也可以放到这个dispatch_once来执行。

然后我们看到它需要一个断言来确定这个代码块是否执行,这个断言的指针要保存起来,相对于第一种方法而言,还需要多保存一个指针。


方法简介中就说的很清楚了:对于在应用中创建一个初始化一个全局的数据对象(单例模式),这个函数很有用。

如果同时在多线程中调用它,这个函数将等待同步等待,直至该block调用结束。

这个断言的指针必须要全局化的保存,或者放在静态区内。使用存放在自动分配区域或者动态区域的断言,dispatch_once执行的结果是不可预知的。




总结:1.这个方法可以在创建单例或者某些初始化动作时使用,以保证其唯一性。2.该方法是线程安全的,所以请放心大胆的在子线程中使用。(前提是你的 dispatch_once_t *predicate_once 对象必须是全局或者静态对象。这一点很重要,如果不能保证这一点,也就不能保证该方法只会被执行一次。)

0 0