MVC 模式是一个非常经典且精简的设计模式,常见的 iOS 项目基本上都采用了这种模式进行开发。 然而,随着项目规模的增长,MVC 模式已经无法很好地帮助我们解耦,构建一个易维护的项目变得愈发艰难。 业务逻辑愈加复杂,导致项目的代码量剧增是最直接的原因。某个 View Controller 可能会有几千行代码,这样的项目代码易读性非常差,维护难度也会增大。而且,这也容易导致数据的读写不同步,在较为混乱的代码中维护复杂的数据流,非常容易导致盘根错节的问题。 MVVM 在 MVC 的基础上增加了 ViewModel,实现了与 MVP 相近的效果。 由于 App 开发往往涉及许多导航逻辑的处理,而这些导航逻辑如果与业务逻辑耦合在一起就会难以管理,MVVM-C 也在iOS开发社区中应运而生。 接下来,请和 Ficow 一起结合架构图来分析这两种App设计模式吧~
MVC 模式是一个非常经典且精简的设计模式,常见的 iOS 项目基本上都采用了这种模式进行开发。 然而,随着项目规模的增长,MVC 模式已经无法很好地帮助我们解耦,构建一个易维护的项目变得愈发艰难。 业务逻辑愈加复杂,导致项目的代码量剧增是最直接的原因。某个 View Controller 可能会有几千行代码,这样的项目代码易读性非常差,维护难度也会增大。而且,这也容易导致数据的读写不同步,在较为混乱的代码中维护复杂的数据流,非常容易导致盘根错节的问题。 MVP 在 MVC 的基础上采用了面向接口/协议编程,最终使业务逻辑代码更易测试。如果结合 Clean Architecture,还可以有效地减轻我们的思维负担。 接下来,请和 Ficow 一起结合架构图来分析这两种App设计模式吧~
| Swift , iOS , 架构 , 测试 , XCTest
如今,项目中的业务逻辑越来越复杂,代码量也疯狂飙升,某个 View Controller 可能会有几千行代码。这些项目的代码易读性非常差,维护难度也比较大,测试的难度更大! 如果需要进行人工测试,而且如果App中的运行时状态非常多,就会非常消耗测试人员。 有什么办法可以解决或者至少缓解一下这个问题吗?目前来看,自动化测试是一个最优解。对于开发人员而言,单元测试是最常见的自动化测试,也是比较有效的测试方法。 然而,想写出易测试的代码需要掌握一定的知识和技巧。接下来,请和 Ficow 一起看看如何基于依赖注入和控制反转构建易测试的代码吧~
| iOS , Xcode , 测试 , Extension
App Extensions 可以有效地扩展应用的使用场景,让用户更便捷地享用我们提供的服务。 某些 App Extensions 相对来说会比较常见,如:Home 页面的小插件(Widget)、分享、富文本推送通知等等。 越来越多的应用开始支持 App Extensions,调试 App Extension 自然必不可少。如何调试并且高效地调试就变成了一个我们需要解决的问题。 可能有朋友会说,在 Scheme 处选中 App Extension,然后直接运行就可以了。 抱歉,这样做是不行的呢!如果您也这样认为,那么 Ficow 建议您去回顾官方文档对 App Extension 的讲解。实际上,App Extension 是不可以作为可执行程序独立运行的,它只能被动地被系统启动。 现在,和 Ficow 来了解一下相关的内容吧~
应用在执行后台任务时,莫名其妙地就崩溃了。登录 Firebase 查看 Crashlytics 控制台,在 crash_info_entry_0 这个 Key 对应的 Value 处看到一个令人困惑的错误:BUG IN CLIENT OF LIBDISPATCH: Unbalanced call to dispatch_group_leave()。 查看了相关的代码,并没有调用 dispatch_group 或相关的方法。好吧,先请强大的 Google 帮帮我~ Google 的搜索结果里面有一条记录是苹果官方的开发者论坛:crash iOS 14 - Unbalanced call to dispatch_group_leave() 其中,GaelPB 的推测观点 是比较接近最终答案的(不过,不正确)。 在此,Ficow 先说明问题的成因: App 在 iOS 14 中执行后台任务时,completionHandler 被多次调用,就会导致崩溃。 这是如何调查出来的呢?问题如何解决呢?希望后文中 Ficow 的 debug 思路能够对您有帮助~
什么是泛型?《Swift 进阶》中给出的定义是:泛型编程是一种可以保持类型安全性的代码重用技术。 泛型也被成为 参数化多态。 实际上,就是让编译器帮我们生成结构重复的代码。 比如,我们要实现最大值函数用于获取两个整数、两个浮点数、两个字符串中较大的值。 如果不借助泛型,我们需要为每一种数据结构单独定义一个 max 函数: func max(_ a: Int, _ b: Int) -> Int { return a > b ? a : b } func max(_ a: Float, _ b: Float) -> Float { return a > b ? a : b } func max(_ a: String, _ b: String) -> String { return a > b ? a : b } 如果借助泛型,我们只需要定义一个 max 泛型函数,然后让每种数据结构遵循 Comparable 协议即可: func max<T>(_ x: T, _ y: T) -> T where T : Comparable { return x > y ? x : y } 泛型通常都会和协议成对出现,因为 Swift 要通过协议明确约束泛型参数的行为。上面的 max 泛型函数就是一个很好的例子,泛型 T 需要遵守 Comparable 协议。 Swift 中的泛型也非常强大,快来深入了解一下吧~