重庆分公司,新征程启航
为企业提供网站建设、域名注册、服务器等服务
这篇文章主要介绍了实际应用 MVVM 的过程中的一些问题和解决方案
我们提供的服务有:网站制作、成都做网站、微信公众号开发、网站优化、网站认证、潜江ssl等。为成百上千企事业单位解决了网站和推广的问题。提供周到的售前咨询和贴心的售后服务,是有科学管理、有技术的潜江网站制作公司
MVVM(Model View ViewModel)是一种 MVC(Model View Controller)的一种变型,来解决 MVC 中庞大复杂的 Controller 难以维护的问题。大致上讲 MVVM 有几个要求:
MVVM 和 MVC 有很多类似的特点,主要的不同有:
另外一点,MVVM 默认 View 和 View Controller 有一个一对一的关系,一般我们把这两个看做一个整体,会以 .swift 文件 和 Storyboard 的形式出现。
View Model 的工作是处理所有的展示数据的逻辑。如果一个 model 中有一个 NSDate 对象, NSDateFormatter 就会在 View Model 中用来设置日期的展示形式。
View Model 不能接触任何用户界面的部分,View Model 文件中不应该 import UIKit ,View Controller 会观察 View Model 去了解什么时候显示新的数据(通过 KVO 或者 FRP(Functional Reactive Programming))
MVVM 和 MVC 有一个共同的弱点:没有清楚的定义应该把网络请求部分放在哪里。在实际操作过程中,我会把网络请求放在 View Model 文件里面,但之后我打算把网络请求放在自己独立的一个类中,View Model 文件会拥有这个对象。
下面我们主要谈一谈实际应用 MVVM 过程中一些挑战:
例如你想构造这样一个常用的界面,有一个 segment control 在屏幕顶部,屏幕的其他部分是一个 collection view,选择不同的 segment,就会展示不同样式的 collection view,元素的排列顺序。我们定义了一个 enum 来枚举所有的排列样式:
那么这个 enum 在 MVVM 模式中应该放在哪里呢?因为这个 enum 决定了数据排列的顺序,每个 cell 中的文字和按钮的 title,这些都属于展示的逻辑,所以这个 enum 看起来应该放在 view model 中。
然而,这些 layout 并不改变要展示的数据,只是决定了要呈现的数据的排列方式和排列顺序,从这个角度上来说 enum 又应该放在 view controller 中。
我的解决方法是把 enum 放在 view model 中,然后在 view model 中加一个对外的 Observable 或者 Signal 来表示使用了哪个 layout,基于用户选择的 segment,view model 更新这个值,然后在 view controller 中根据相应的 layout 改变 collection view 的样式,view controller 也可以根据这个值来决定用哪个 cell reuse identifier
iOS 开发者在用 MVVM 和 FRP 写应用的时候最常见的问题可能就是 ViewModel 怎么把数据展现给 ViewController。当 Model 层的数据发生变化更新的时候,ViewController 需要得到通知然后做出相应的 UI 更新,我们一般会用到两种机制:
第一个选项很吸引人,因为可以在 View Controller 中决定怎么选择观察那些 property。然而,我不推荐在 Swift 中使用第一个选项,因为 Swift 在 KVO 中没有类型检查,你需要对 AnyObject 强制转换类型很多次。
第二个选项是比较 Swift 的方式,基于 Swift 的 generics 特性,signals,sequences,observables 可以支持编译过程中的类型检查。
但有时候在 view model 增加这些 Signals 或者 Observables 有些困难。Swift 的初始化方法对于什么时候对 property 赋值有非常明确的规定。Signals 或者 Observables 需要使用 view model 内部的状态,所以它们必须在 super.init() 之后才能创建,但是另一方面,我们在调用 super.init() 之前保证所有 property 已经被赋值了,包括那些 Signal/Observable property。
这是个先有鸡还是先有蛋的问题。
我采用比较简单的解决方法:定义成 var 的隐式可选类型,这样就可以在 super.init() 之后才给 property 赋值。这不是一个完美的解决办法。我们可以用 lazy var property 的闭包赋值来代替上面的方法。在 Swift 不断完善和更新的过程中,大家也可以探索其他更好的办法。
举一个很常用的例子,用户点击 collection view 中的一个 cell,跳转到详情页面。用户点击的操作应该在 view controller 中处理,具体内容是展现一个新的详情页面。但是 view controller 不能直接接触 models,我们要如何用 MVVM 模式实现这样的用户交互呢?
我的解决方案是利用 Swift 的闭包。首先在 view model 中定义一个闭包:
然后在 view model 中添加一个 property:
接着我需要调用闭包,在 view model 中定义一个view controller 可以调用的函数,这个函数的参数是可以决定使用什么数据,一般情况下常用 index path:
现在当用户选中一个 cell,会调用 view model 中的这个函数,并且传入 index path 参数,view model 决定使用哪个数据,并调用在 view controller 中定义的闭包,例如:
最后一个问题是怎么创建这个 view model。我们需要传递一个闭包给view model 的初始化函数,然后用 lazy loading 来调用 view model 的初始化函数。
1.好处:
能快速的遍历数组等容器对象
2.实质:
都是对容器里的每一个元素的遍历
3.注意事项:
为了增加代码的可读性,避免不必要的错误,尽量让forin的前部分的类型和数组里的元素类型相同,后面是要遍历的容器str第一次循环对应第一个元素
实例:
Student *stu1=[[Student alloc]initWithName:@"商帅"];
Student *stu2=[[Student alloc]initWithName:@"刘珊珊"];
Student *stu3=[[Student alloc]initWithName:@"杨林"];
Student *stu4=[[Student alloc]initWithName:@"腾飞"];
NSArray *arr=@[stu1,stu2,stu3,stu4];
//第一次循环前面保存的是第一个数组
//遍历数组里每一个学生的姓名
//因为Student的类型里没有数组了,只有name了所以不用二次遍历
for (Student *stu in arr) {
NSLog(@"%@",stu.name );
}
1.枚举的变量名可读性高, 可以代替原本没有可读性的1,2,3...,
这样写代码,和后期阅读代码, 都很方便.
2.现在Swift对枚举提供了更多灵活的处理支持.
如果使用Swift开发的话,更能体会到枚举带来的方便快捷.
前言
Enum,也就是枚举,从C语言开始就有了,C++、Java、Objective-C、Swift这些语言,当然都有对应的枚举类型,功能可能有多有少,但是最核心的还是一个—规范的定义代码中的状态、选项等“常量”。
举个例子,我们想要在开发中根据四季的变化做不同的操作,我们首先想到的可能会是定义一个int型变量,通过为该变量赋不同的`值来表示四季(e.g. 1为春天,2为夏天,3为秋天,4为冬天).但是,对于阅读该份代码的人来讲,他们并不能在短时间内就明白数字1234所代表的具体含义.这个时候,选择使用枚举则可以应对该需求
C语言中的枚举
在C语言中定义枚举类型的方式如下
enum 枚举名称{ 标识符 = 整型常量, 标识符 = 整型常量, 标识符 = 整型常量, 标识符 = 整型常量};
以"定义一个枚举名称为Test,拥有TestA、TestB等枚举成员的枚举类型"为例,定义方式如下
enum Test{ TestA = 0, TestB = 1, TestC = 2, TestD = 3};
注1: 当枚举成员都未设置"=整型常量"时,默认从第一个标识符开始,从0依次递增加1
注2: 当为某个枚举成员设置"=整型常量"时,其后面的标识符会在此基础上依次递增加1
在使用该枚举类型时,我们需要创建一个枚举变量
enum 枚举名称 枚举变量 = 枚举变量值;
以"创建一个枚举名称为Test的枚举变量test,并为其赋值TestB"为例
enum Test test = TestB;
注: 枚举变量值的取值只能从对应枚举类型的枚举成员中选择
我们也可以在定义枚举类型的同时定义枚举变量,该枚举变量不用初始化,可直接对其进行赋值并使用
enum Test{ TestA = 0, TestB, TestC, TestD} test;
注: 该枚举名称可以省略,直接定义枚举变量即可(称作"匿名枚举")
通过typedef为枚举类型设置一个"别名",这样便可以像使用int一样使用枚举类型
typedef enum _Test{ TestA = 0, TestB, TestC, TestD} Test;
定义"别名"后,便可采用如下方式定义枚举变量
Test test = TestB;
针对匿名枚举,在通过typedef设置"别名"便变成如下形式
typedef enum{ TestA = 0, TestB, TestC, TestD} Test;
定义"别名"后,便可采用如下方式定义枚举变量
Test test = TestB;
OC语言中的枚举
在Objective-C语言中,Apple在iOS6中引入了两个宏来重新定义枚举类型(即:NS_ENUM与NS_OPTIONS),这两者在本质上并没有差别,都是用于定义枚举类型,但是在使用中NS_ENUM多用于一般枚举,而NS_OPTIONS则多用于带有移位运算的枚举
NS_ENUM使用示例
typedef NS_ENUM(NSInteger, Test){ TestA = 0, TestB, TestC, TestD};
NS_OPTIONS使用示例
typedef NS_OPTIONS(NSUInteger, Test) { TestA = 1 0, TestB = 1 1, TestC = 1 2, TestD = 1 3};
typedef NS_OPTIONS(NSUInteger, Test) { TestNone = 0, TestA = 1 0, TestB = 1 1, TestC = 1 2, TestD = 1 3};
带有移位运算的枚举多用于同一个枚举变量可以同时赋值多个枚举成员的情况,其原理就是将各个枚举值按位或(|),因为移位运算的枚举成员可以保证按位或(|)计算之后结果的唯一性,所以每种结果都能反向计算出是由哪几个枚举成员按位或(|)而成
以"使用按位或(|)为枚举变量test同时赋值枚举成员TestA、TestB与TestC"为例
Test test = TestA | TestB;test |= TestC;
以"使用按位异或(^)为枚举变量test去掉一个枚举成员TestC"为例
Test test = TestA | TestB | TestC;test ^= TestC;
注: 如果枚举变量test本身并未赋值TestC,那么使用按位异或(^)会为枚举变量test多赋值一个枚举成员TestC
以"使用按位与()判断枚举变量test是否赋值了枚举成员TestA"为例
Test test = TestA | TestB;if (test TestA){ NSLog(@"yes");}else{ NSLog(@"no");}
总结
以上就是这篇文章的全部内容了,希望能对大家的学习或者工作带来一定的帮助,如果有疑问大家可以留言交流。