我们经常在编程中使用各种设计模式,在iOS中比较常见的设计模式有:单例模式、委托模式、观察者模式,当然实际上在Cocoa和Cocoa Touch框架中不仅仅是设计到这些设计模式,还有很多设计模式以及常规设计模式的变种等等。那什么是设计模式呢?
在软件工程中,设计模式(design pattern)是对软件设计中普遍存在(反复出现)的各种问题,所提出的解决方案。这个术语是由埃里希·伽玛(Erich Gamma)等人在1990年代从建筑设计领域引入到计算机科学的。(引自维基百科)
设计模式(Design pattern)是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码可靠性。 毫无疑问,设计模式于己于他人于系统都是多赢的;设计模式使代码编制真正工程化;设计模式是软件工程的基石脉络,如同大厦的结构一样。(引自百度百科)
通过解释我们可以基本了解设计模式是什么这个问题,当然我们不在这里深入探讨设计模式概念理论方面的问题,而是对我们在iOS开发中常用到的几种设计模式进行梳理,以方便日常开发。
Tips:设计模式和架构模式通常容易混淆
- 不同点 设计模式是解决某类常见问题的方法思路总结(尺度小) 框架模式是从软件开发总体结构上提出的一中框架结构(尺度大)
- 共同点 降低耦合度、调高可拓展性、易维护
比如:MVC是架构模式,单例是设计模式
1. 单例模式(Singleton)
什么是单例
在程序的整个生命周期内,保证类只有一个实例对象并可在全局调用。
如何实现单例
实现单例我们需要注意两点:1.我们需要保证单例被实例化一次之后,不能被重复实例化2.在多线程模式下避免同时被一个以上的线程调用实例化方法而造成多个实例。
实现单例:
@synchronized同步锁
static SingletonInstant *_single = nil;
@implementation SingletonInstant
+ (instancetype)sharedInstant {
if (!_single) { // 判断放在加锁之前防止多次加锁
@synchronized(self) { // 添加同步锁防止保证线程安全
_single = [[super allocWithZone:NULL] init];
}
}
return _single;
}
// 上面使用allocWithZone不使用alloc是因为alloc方法会调用
// allocWithZone方法,如果使用alloc还要在重载的allocWithZone中
// 在实现一次判断和加锁,因此为了简化代码,
// 直接使用allocWithZone,在重载的方法中只要调用一次单例方法就可以了
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
return [self sharedInstant];
}
// copy方法也会调用copyWithZone方法,因此直接返回self就好
- (id)copyWithZone:(NSZone *)zone {
return self;
}
@end
#pragma - mark MRC实现
static SingletonInstant *_single = nil;
+ (SingletonInstant*)sharedManager
{
if (_single == nil) {
@synchronized (self) {
_single = [[super allocWithZone:NULL] init];
}
}
return _single;
}
+ (id)allocWithZone:(NSZone *)zone
{
return [[self sharedManager] retain];
}
- (id)copyWithZone:(NSZone *)zone
{
return self;
}
// 比ARC多出来的关于内存管理的几个方法
- (id)retain
{
return self;
}
- (NSUInteger)retainCount
{
return NSUIntegerMax; //denotes an object that cannot be released
}
- (void)release
{
//do nothing
}
- (id)autorelease
{
return self;
}
更多关于alloc和allocWithZone的问题请查阅参考4
,MRC实现查阅参考3
GCD加锁
使用 dispatch_once_t 保证线程安全
static SingletonInstant *_single = nil;
@implementation SingletonInstant
+ (instancetype)sharedInstant {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
if (!_single) {
_single = [[super allocWithZone:NULL] init];
}
});
return _single;
}
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
return [self sharedInstant];
}
- (id)copyWithZone:(NSZone *)zone {
return self;
}
@end
在dispatch_once方法中的代码在程序的真个声明周期只会执行一次。,关于GCD请自行学习。
MRC不再详述
单例的实现基本雷同,在网上也有一些大同小异的其他写法,只要掌握单例思想,任何一种实现都是可以的,差异在细节上,孰优孰劣任君斟酌。