| |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
-> 移动开发 -> iOS 单例模式详解/避免滥用单例 -> 正文阅读 |
|
[移动开发]iOS 单例模式详解/避免滥用单例 |
前言hihi勇敢的小伙伴儿们大家好,单例相信大家都很熟悉了,它在我们的开发中给到了不少帮助,但是你真的熟悉它吗?是否只是一个泛泛之交,而没有走进它的内心世界呢? 今天,我们一起走近“单例”,偷偷地了解一下单例吧~ 正文一、单例介绍单例模式:单例模式,属于创建类型的一种常用的软件设计模式。通过单例模式的方法创建的类在当前进程中只有一个实例。 附上23种设计模式: ??????? ?为了我们能更好的理解单例模式,我列举以下几个cocoa框架中常用的单例: 1.UIApplication:应用程序。一个UIApplication对象就代表着一个应用程序,每个应用程序有且仅有一个UIApplication对象,开发中最常用的是使用它的openURL函数来跳转到其他应用程序,通过 [UIApplication sharedApplication] 类方法可以获得。 2.NSNotificationCenter:通知中心。iOS中的通知中心是一种消息广播,采用观察者模式和单例模式,一个应用有且仅有一个通知中心。通过 [NSNotificationCenter defaultCenter] 类方法可以获得。 3.NSFileManager:文件管理器。它是iOS文件系统的接口,用来创建、修改、访问文件。一个应用有且仅有一个文件管理器。通过 [NSFileManager defaultManager] 类方法可以获得。 4.NSUserDefaults:用户偏好设置。它主要用来存储简单的键值对数据,数据持久化最简单和基础的一种方案。通过 [NSUserDefaults standardUserDefaults] 类方法可以获得。 5.NSURLCache:URL缓存。通过将NSURLRequest对象映射到NSCachedURLResponse对象来实现对URL加载请求的响应的缓存。通过 [NSURLCache sharedURLCache] 类方法可以获得。 1.1 单例模式的要点1.只能有一个实例; 2.它必须自行创建这个实例; 3.它必须自行向整个系统提供这个实例。 从具体实现角度来说,是以下三点: 1.单例模式的类只提供私有的构造函数; 2.类定义中含有一个该类的静态私有对象(实例); 3.提供一个静态的公有函数用于创建或获取它本身的静态私有对象(实例)。 1.2 单例模式的优点1.实例控制:单例可以保证系统中该类有且仅有一个实例,确保所有对象都访问这个唯一实例; 2.灵活性:因为类控制了实例化的过程,所以类可以灵活更改实例化过程; 3.节省开销:因为只有一个实例,所以减少内存开发和系统的性能开销。 1.3 单例模式的缺点1.由于单例模式中没有抽象层,可扩展性比较差。 2.实例一旦被创造,对象指针保存在静态区,那么在堆区分配的空间只有在App结束后才会被释放; 3.单例类职责过重,在一定程度上违背了“单一职责原则”。 4.滥用单例会带来一些负面问题,比如,单例会隐性地让毫不相关的类产生耦合等问题。(详见3.1) 1.4 单例的生命周期首先我们复习一下内存的五大存储区域: 由于程序里,一个单例类只能初始化一次,为了保证它在使用中始终存在,所以单例对象一旦建立,对象指针保存在静态区,单例对象(实例)在堆中分配的内存空间,只有应用程序终止后才会被释放。 那么问题来了,如果单例的实例被置为nil,内存是否会得到释放吗?
当单例类的实例对象赋值 nil 后,触发了单例的 dealloc方法。 由于静态变量修饰的指针保存在了全局区域,所以不会被释放,但是指针保存的关联对象地址是alloc申请的内存,在堆中,内存会因引用计数为0被系统调用?dealloc 方法释放掉。(我理解的不知道对不对) 二、单例的实现单例的实现重点就是防止在外部调用的时候出现多个不同的实例,也就是说要从创建的方式入手禁止出现多个不同的实例。 主要做到以下几点: 1.防止调用 [[A alloc] init] 引起错误 2.防止调用 new 引起错误 3.防止调用 copy 引起错误 4.防止调用 mutableCopy 引起错误 2.1 典型的单例写法
缺点:无法保证多线程情况下只创建一个对象。适用于只有单线程。 2.2 加锁的写法
2.3 GCD写法【常用】2.3.1 重写父类方法
dispatch_once 主要是根据 onceToken 的值来决定怎么去执行代码。 1.当 onceToken = 0 时,线程执行 dispatch_once 的 block 中代码; 2.当?onceToken = -1 时,线程跳过 dispatch_once 的 block 中代码不执行; 3.当 onceToken 为其他值时,线程被阻塞,等待 onceToken 值改变。 当线程调用 shareInstance,此时?onceToken = 0,调用 block 中的代码,此时?onceToken = 其他值。当其他线程再调用 shareInstance 方法时,onceToken为其他值,线程阻塞。当 block 线程执行完 block 之后,onceToken = -1,其他线程不再阻塞,跳过 block。下次在调用 shareInstance 时, block 已经为 -1,直接跳过 block。 2.3.2?禁止外部调用 这种写法还蛮简单好用的~
当运行 init 或者 new 时,会报错 'init' is unavailable 或者 'new' is unavailable。 2.3.3 宏定义写法 写成宏是比较方便也比较常见的一种:
用法也很简单:
2.4 免锁写法
1,2,4三种写法还需将 init new 重写或禁用才算完整哦~ 三、单例的滥用3.1 全局状态大多数的开发者都认同使用全局可变的状态是不好的行为。太多状态使得程序难以理解,难以调试。我们这些面向对象的程序员在最小化代码的状态复杂程度的方面,有很多需要向函数式编程学习的地方。(这段话挺费解的)
在上面这个简单的数学库的实现中,程序员需要在调用?
当为调用? 把下面的代码和上面的例子做对比:
这里,对变量? 那么,这个例子和单例又有什么关系呢?用 Mi?ko Hevery 的话来说,???????"单例就是披着羊皮的全局状态"。一个单例可以被使用在任何地方,而不需要显式地声明依赖。就像变量?
在上面的例子中, 让我们来看一个更具体的例子,并且暴露一个使用全局可变状态的额外问题。比如我们想要在我们的应用中构建一个网页查看器。为了支持这个查看器,我们构建了一个简单的 URL cache:
这个开发者开始写一些单元测试来保证代码在一些不同的情况下都能达到预期。首先,他写了一个测试用例来保证网页查看器在设备没有连接时能够展示出错误信息。然后他写了一个测试用例来保证网页查看器能够正确的处理服务器错误。最后,他为成功情况时写了一个测试用例,来保证返回的网络内容能够被正确的显示出来。这个开发者运行了所有的测试用例,并且它们都如预期一样正确。赞! 几个月以后,这些测试用例开始出现失败,尽管网页查看器的代码从它写完后就从来没有再改动过!到底发生了什么? 原来,有人改变了测试的顺序。处理成功的那个测试用例首先被运行,然后再运行其他两个。处理错误的那两个测试用例现在竟然成功了,和预期不一样,因为 URL cache 这个单例把不同测试用例之间的 response 缓存起来了。 持久化状态是单元测试的敌人,因为单元测试在各个测试用例相互独立的情况下才有效。如果状态从一个测试用例传递到了另外一个,这样就和测试用例的执行顺序就有关系了。有 bug 的测试用例,尤其是那些本来不应该通过的测试用例,是非常糟糕的事情。 3.2 对象的生命周期另外一个关键问题就是单例的生命周期。当你在程序中添加一个单例时,很容易会认为 “永远只会有一个实例”。但是在很多我看到过的 iOS 代码中,这种假定都可能被打破。 比如,假设我们正在构建一个应用,在这个应用里用户可以看到他们的好友列表。他们的每个朋友都有一张个人信息的图片,并且我们想使我们的应用能够下载并且在设备上缓存这些图片。 使用?
我们继续构建我们的应用,一切看起来都很正常,直到有一天,我们决定去实现“注销”功能,这样用户可以在应用中进行账号切换。 突然我们发现我们将要面临一个讨厌的问题:用户相关的状态存储在全局单例中。当用户注销后,我们希望能够清理掉所有的硬盘上的持久化状态。否则,我们将会把这些被遗弃的数据残留在用户的设备上,浪费宝贵的硬盘空间。对于用户登出又登录了一个新的账号这种情况,我们也想能够对这个新用户使用一个全新的? 问题在于按照定义单例被认为是“创建一次,永久有效”的实例。你可以想到一些对于上述问题的解决方案。或许我们可以在用户登出时移除这个单例:
这是一个明显的对单例模式的滥用,但是它可以工作,对吧? 我们当然可以使用这种方式去解决,但是代价实在是太大了。我们不能使用简单的的? 现在我们需要对使用缩略图 cache 的代码的执行顺序非常小心。假设当用户正在执行登出操作时,有一些后台任务正在执行把图片保存到缓存中的操作:??????????????
我们需要保证在所有的后台任务完成前,? 由于对于单例实例来说它没有明确的所有者(因为单例自己管理自己的生命周期),“关闭”一个单例变得非常的困难。 分析到这里,我希望你能够意识到,“这个缩略图 cache 从来就不应该作为一个单例!”。问题在于一个对象得生命周期可能在项目的最初阶段没有被很好得考虑清楚。举一个具体的例子,Dropbox 的 iOS 客户端曾经只支持一个账号登录。它以这样的状态存在了数年,直到有一天我们希望能够同时支持多个用户账号登录 (同时登陆私人账号和工作账号)。突然之间,我们以前的的假设“只能够同时有一个用户处于登录状态”就不成立了。如果假定了一个对象的生命周期和应用的生命周期一致,那你的代码的灵活扩展就受到了限制,早晚有一天当产品的需求产生变化时,你会为当初的这个假定付出代价的。 这里我们得到的教训是,单例应该只用来保存全局的状态,并且不能和任何作用域绑定。如果这些状态的作用域比一个完整的应用程序的生命周期要短,那么这个状态就不应该使用单例来管理。用一个单例来管理用户绑定的状态,是代码的坏味道,你应该认真的重新评估你的对象图的设计。 3.3 避免使用单例既然单例对局部作用域的状态有这么多的坏处,那么我们应该怎样避免使用它们呢? 让我们来重温一下上面的例子。既然我们的缩略图 cache 的缓存状态是和具体的用户绑定的,那么让我们来定义一个user对象吧:
我们现在用一个对象来作为一个经过认证的用户会话的模型类,并且我们可以把所有和用户相关的状态存储在这个对象中。现在假设我们有一个 view controller 来展现好友列表:???????
我们可以显式地把经过认证的 user 对象作为参数传递给这个 view controller。这种把依赖性传递给依赖对象的技术正式的叫法是依赖注入,它有很多优点:??????? 1.对于阅读这个?SPFriendListViewController?头文件的读者来说,可以很清楚的知道它只有在有登录用户的情况下才会被展示。
就算后台任务还没有完成,应用其他地方的代码也可以创建和使用一个全新的 SPUser 对象,而不会在清理第一个实例时阻塞用户交互。 为了更详细的说明一下第二点,让我们画一下在使用依赖注入之前和之后的对象图。 假设我们的? view controller 自己,以及自定义的 image view 的列表,都会和? 这里的问题在于这个好友列表的 view controller 可能仍然在执行代码 (由于后台操作的原因),并且可能因此仍然有一些没有执行的涉及到? 和使用依赖注入的解决方案对比一下: 简单起见,假设? 这个对象图看起来和使用单例时很像。那么,区别是什么呢? 关键问题是作用域。在单例那种情况中, 当用户登录一个新账号,我们应该能够构建并且与全新的? 3.4 结论希望这篇文章中的内容读起来不像奇幻小说那样难以理解。人们已经对单例的滥用抱怨了很多年了,并且我们也都知道全局状态是很不好的事情。但是在 iOS 开发的世界中,单例的使用是如此的普遍以至于我们有时候忘记了我们多年来在其他面向对象编程中学到的教训。 这一切的关键点是,在面向对象编程中我们想要最小化可变状态的作用域。但是单例却因为使可变的状态可以被程序中的任何地方访问,而站在了对立面。下一次你想使用单例时,我希望你能够好好考虑一下使用依赖注入作为替代方案。 写在最后上面关于避免滥用单例的部分,写的太好了,所以我把全文搬运过来了,看语法应该是外文翻译过来的,条理清晰,就是翻译成中文后读起来有些地方会有些晦涩,今天看到这篇文章虽收益颇丰,但是似乎并没有什么真正的收获,不知道在自己的代码上该如何运用,在设计程序之前考虑对象的生命周期等等经验真的很宝贵!所以在这里分享出来,希望更多人看到和学习到! 希望对你们也有所帮助! 如果有错误还请小伙伴儿们指正,感激不尽~ ???????参考文章: 单例模式百度百科:单例模式_百度百科 iOS 单例:https://www.cnblogs.com/dins/p/ios-singleton.html OC中单例的各种写法:OC中单例的各种写法及基本讲解 - 走看看 |
|
移动开发 最新文章 |
Vue3装载axios和element-ui |
android adb cmd |
【xcode】Xcode常用快捷键与技巧 |
Android开发中的线程池使用 |
Java 和 Android 的 Base64 |
Android 测试文字编码格式 |
微信小程序支付 |
安卓权限记录 |
知乎之自动养号 |
【Android Jetpack】DataStore |
|
上一篇文章 下一篇文章 查看所有文章 |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 | -2024/11/25 3:58:58- |
|
网站联系: qq:121756557 email:121756557@qq.com IT数码 |