(一二三)基于GCD的dispatch_once实现单例设计

要实现单例,关键是要保证类的alloc和init仅仅被调用一次。而且被自身强引用防止释放。

近日读唐巧先生的《iOS开发进阶》。受益匪浅,通过GCD实现单例就是收获之中的一个,以下把这种方法与大家分享。

在GCD中,有一个函数dispatch_once,能够实现代码段的一次性运行,和static修饰的变量赋值的一次性一样。我们结合static和dispatch_once,就能够简单的实现单例。

以下的代码实现了SomeClass单例:

#import <Foundation/Foundation.h>

@interface SomeClass : NSObject

+ (SomeClass *)sharedInstance;

@end
#import "SomeClass.h"

@implementation SomeClass

+ (SomeClass *)sharedInstance{

    static id sharedInstance = nil;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        sharedInstance = [[self alloc] init];
    });

    return sharedInstance;

}

@end

以下解释一下这段代码。

第一句新建一个sharedInstance静态强指针,是为了指向创建好的单例,防止其释放,仅仅有第一次进入的时候指针被赋值为nil。

一定注意dispatch_once_t变量必须是静态,它的值用于推断是否已经运行一次。

第二句和dispatch_once是固定使用方法,这样能够实现block内的代码一次性运行。也就是说仅仅有第一次调用这种方法时才会实例化类。之后都是返回指针指向的值。

最后返回指针,就相当于拿到了单例。

对单例的执行结果进行验证:

我们多次获取单例对象而且打印地址,能够发现地址是一样的。

#import "ViewController.h"
#import "SomeClass.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];

    SomeClass *sc1 = [SomeClass sharedInstance];
    SomeClass *sc2 = [SomeClass sharedInstance];
    SomeClass *sc3 = [SomeClass sharedInstance];
    NSLog(@"%p %p %p",sc1,sc2,sc3);

}

@end
2015-08-17 20:59:22.139 基于GCD实现单例[2785:31918] 0x7fb40af11b90 0x7fb40af11b90 0x7fb40af11b90

通过这样的方式,简单高效的实现了单例,值得使用。

时间: 2024-10-11 23:26:35

(一二三)基于GCD的dispatch_once实现单例设计的相关文章

使用GCD的dispatch_once创建单例

使用GCD的dispatch_once创建单例 介绍了创建单例的方式,不过后来发现其实在ios 4.0后有更加简单的方式. 就是使用GCD的功能 代码如下: + (instantClass *)sharedClient { static instantClass *_sharedClient = nil; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ _sharedClient = [[instantClass al

利用GCD的dispatch_once创建单例

1 + (id)sharedWhatever 2 { 3 static dispatch_once_t pred; 4 static Whatever *whatever = nil; 5 dispatch_once(&pred, ^{ 6 whatever = [[self alloc] init]; 7 }); 8 return whatever; 9 } 该方法有很多优势:   1.线程安全   2.很好满足静态分析器要求   3.和自动引用计数(ARC)兼容   4.仅需要少量代码 该方

利用dispatch_once创建单例

利用dispatch_once创建单例 使用Objective-C实现单例模式的最佳方式向来有很多争论,开发者(包括Apple在内)似乎每几年就会改变他们的想法.当Apple引入了Grand Central Dispatch (GCD)(Mac OS 10.6和iOS4.0),他们也引入了一个很适合用于实现单例模式的函数. 该函数就是dispatch_once: void dispatch_once( dispatch_once_t *predicate, dispatch_block_t bl

IOS基础学习日志(七)利用dispatch_once创建单例及使用

自苹果引入了Grand Central Dispatch (GCD)(Mac OS 10.6和iOS4.0)后,创建单例又有了新的方法,那就是使用dispatch_once函数,当然,随着演进的进行,还会有更多的更好的方法出现.今天就来简要介绍下如何利用dispatch_once创建单例. 在开发中我们会用到NSNotificationCenter.NSFileManager等,获取他们的实例通过[NSNotificationCenter defaultCenter]和[NSFileManage

IOS 使用dispatch_once 创建单例

+ (instantClass *)sharedClient { static instantClass *_sharedClient = nil; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ _sharedClient = [[instantClass alloc] init]; }); return _sharedClient; } IOS 使用dispatch_once 创建单例,码迷,mamicode.co

ios oc 和 swfit 用dispatch_once 创建单例

网上已经有方法了,我这里就是抄了下,原文链接 http://bj007.blog.51cto.com/1701577/649413 http://blog.csdn.net/u010124617/article/details/28856711?utm_source=tuicool 这里的dispatch_once_t就是一个用于标记的整型,没什么特殊之处. 用dispatch_once实现单例的最大好处就是线程安全!!这个经常会被面试问到,建立数据库和网络操作的单例时,确实存在线程安全问题.

Java设计模式中的单例设计

/** * 单例设计模式 * 应用场合:只需要一个对象的 * 作用:保证整个应用程序中某个实例有且只有一个 * 类型有:饿汉模式.懒汉模式 * 下面的例子是一个饿汉模式的例子 */ class SingleDemo { // 1.将构造方法私有化,不允许外部直接创建使用 private SingleDemo() {} // 2.创建类的唯一实例,使用private static修饰 private static SingleDemo instance = new SingleDemo(); //

Spring容器-ApplicationContext的单例设计

Spring容器-ApplicationContext的单例设计 每次通过new创建一个ApplicationContext容器,都会执行refresh方法,看源代码了解到这个refresh方法会重新加载配置文件,并且这个创建的容器对象持有一个所有singleton类型bean的map集合,从而实现单例,而且这个map对象的生命周期和容器对象的生命周期是一样的 如果我们每次都通过new一个容器对象,那么每次都要重新加载配置文件,都要重新生成这个singleton bean的集合,这样所谓的单例就

【小白的java成长系列】——构造方法私有化(单例设计)

有了解过spring框架的童鞋们就知道,spring的bean默认是什么形式呀?---单例形式的. 问:那什么叫做单例?单例其实就是Singleton,顾名思义就是只有单个的实例对象操作. 那为什么要使用单例呢? 至于这个问题,后面再做解释,我们先看代码: package me.javen.oop; public class SingletonDemo { public static void main(String[] args) { Singleton singleton1 = Single