以前写OC中的单例很固定,一直都这样写,后来我就把它放在快捷代码块里面,只要输入singleton就直接输出这段代码
1 2 3 4 5 6 7 8 9 + (instancetype)instance { static Class *sharedInstance = nil; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ sharedInstance = [[self alloc] init]; }); return sharedInstance; }
一直这样写,并没有关注内部实现 这两天一直在复习C++,然后看到了C++的单例模式,发现内部实现还是挺有讲究的
先来看OC中的 这个方法刚学OC的时候就记下来了
1 void dispatch_once( dispatch_once_t *predicate, dispatch_block_t block);
官方文档的解释是,这个方法让block里面的代码在整个应用的生命周期里只执行一次,如果多线程同时调用,这个函数会让线程在block执行完之前都会同步地等待。只执行一次,顺带帮我们解决多线程问题,这是苹果帮我们封装好的,官方也建议用来写单例的方法!!
然后来看C++的
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 Class Singleton { private : Singleton (); static Singleton* instance; public : static Singleton* getInstance { if (instance == NULL ) instance = new Singleton (); return instance; } }
首先,我们要把构造方法设置private,让别人无法调用构造方法 C++的静态对象和方法就像OC中的类的类对象和方法,类似在OC中是以+开头的方法 把instance和getInstance方法都设为静态,让类直接调用
这是一种实现方法,但是并没有考虑到线程安全 在getInstance()方法应该上锁
1 2 3 4 5 6 7 static Singleton* getInstance { lock(); //这里不实现上锁方法,只是探讨单例模式 if(instance == NULL) instance = new Singleton(); unlock(); //伪 return instance; }
这样是可行的,但是每次通过getInstance方法都会加上一个同步锁,加锁是一个非常耗时的操作,可以在没有必要的时候应该尽量避免。
进一步完善,当instance还没有创建的时候才需要加锁,等第二个线程上锁的时候再判断一次是否已经创建了instance
1 2 3 4 5 6 7 8 9 static Singleton* getInstance { if(instance == NULL){ lock(); //伪 if(instance == NULL) instance = new Singleton(); unlock(); //伪 } return instance; }
另外,要保证只能从单例方法获取单例,不能赋值和拷贝 所以要声明赋值和拷贝方法,但是不实现
1 2 Singleton(const Singleton&){}; Singleton&operator=(const Singleton&){}; 在C++11也可以这样
1 2 Singleton(const Singleton&) = delete; Singleton&operator=(const Singleton&) = delete;
所以这是一种比较好的实现方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 Class Singleton { private: //构造函数 Singleton(); //静态对象 static Singleton* instance; //赋值和拷贝方法 Singleton(const Singleton&){}; Singleton&operator=(const Singleton&){}; public: static Singleton* getInstance { if(instance == NULL){ lock(); //伪 if(instance == NULL) instance = new Singleton(); unlock(); //伪 } return instance; } }
关于类的静态成员 在C++中,静态成员不属于类的任何一个对象,所以它们并不是在创建类的对象时被定义的,它们不是由类的构造函数初始化的,一般来说不能再累的内部初始化静态成员,必须在类的外部定义和初始化每个静态成员,类似于全局变量,静态成员定义在任何函数之外,一旦被定义,将一直存在于成员的整个生命周期中 我们可以利用整个特性来写单例模式
1 2 3 4 5 6 7 8 9 10 11 12 class Singleton { private: Singleton(); static Singleton* instance; public: static Singleton* getInstance{ return instance; } } //然后在类外初始化 Singleton* Singleton::instance = new Singleton();
这种方法更加简洁,并且也是线程安全的,因为除了函数里面的静态变量,其他的静态变量都是在main函数之前就创建好了,单线程创建的,参考stackoverflow上的一个回答
在C#中可以直接在类内初始化
1 2 3 4 5 6 7 8 9 10 public sealed class Singleton { private Singleton(){} private static Singleton instance = new Singleton(); public static Singleton getInstance { get { return instance; } } }
这种方法比上一种二重锁的方法性能要更好 在C#中还有更好的方法
1 2 3 4 5 6 7 8 9 10 11 12 public sealed class Singleton { Singleton(){} public static Singleton getInstance{ get { return Nested.instance; } } Class Nested{ static Nested(){} internal static readonly Singleton instance = new Singleton(); } }
嵌套类,当用到Nested类的时候才会调用构造方法创建instance 由于无法实现类内初始化instance,所以这种方法我还不知道C++怎么实现 以上方法都是参考自剑指Offer