设计模式-MVC模式和KVC/KVO模式以及单例模式的理解记录

MVC模式:

M: model 是数据模型
V: view 是视图
C: controller 是控制器

一张图解释MVC

model 与view 不能直接交互,需要通过controller来同步
model负责数据和状态的更新,view显示给用户看的内容

如何通信:

  • controller到model: 导入model类,实例化model对象,进行数据同步
    controller到view: 在controller中创建View的控件outlet属性进行通信

  • view到controller: 不能在view类中实例化controller,需要使用target: action目标动作机制或者delegate委托机制实现通信
    target:action:用户在view上触发事件,view会产生一个action动作,在controller中,通过addtarget:方法,接受action动作。controller自身设置target,view在需要通知controller时向controller发送action
    delegate: controller作为view的被委托者,代理。为view 提供显示需要的数据
    分为动作类delegate:在controller中响应action动作和data source数据源类delegate:为view 提供需要显示的数据

  • model到controller: 不能在model类中实例化controller,因为model不知道有多少个controller引用它。
    需要使用通知机制和KVO模式和controller通信
    Notification: model自己设置一个通知中心NSNotificationCenter,需要知道model数据变化的的controller自己注册一个通知addObserver来监听model的数据数据变化,让不需要监听model需要移除注册监听
    KVO: 使用addObserver:forKeyPath:option:context:方法和removeObserver:forKeyPath:方法来对model设置和移除注册监听

MVC模式的优势:

  1. 低耦合性
  2. 高重用性
  3. 可适用性
  4. 可维护性

MVVM的简单理解:

MVVM模式是MVC模式的增强,把MVC中controller里的表示model给view显示数据的表示逻辑部分提取出来了——view model
MVVM 即 model view viewcontroller view model

单例模式:

应用程序中的类只拥有一个实例:
NSApplication NSFileManager NSBundle UserDefault VIAccelermeter
单例易于访问

实现单例:

  1. 在类的内部提供一个static修饰的全局变量
  2. 提供一个类方法,方便外界访问
  3. 重写+allocWithZone方法,保证永远都只为单例对象分配一次内存空间
  4. 重写-copyWithZone方法和-MutableCopyWithZone方法

KVC模式:

KVC概念::KeyValueCoding 键值编码

KVC可以直接通过字符串类型的属性名key来访问某个类属性的机制,支持对象和基本数据类型(自动封装,解装)不是通过调用的Setter、Getter方法访问
关键方法定义在 NSKeyValueCodingProtocol

KVC用法:

获取值

- valueForKey:,传入NSString属性的名字。
- valueForKeyPath:,传入NSString属性的路径,xx.xx形式。
- valueForUndefinedKey它的默认实现是抛出异常,可以重写这个函数做错误处理。

修改值

- setValue:forKey:
- setValue:forKeyPath:
- setValue:forUndefinedKey:
- setNilValueForKey: 当对非类对象属性设置nil时,调用,默认抛出异常。
一对多关系成员的情况
- mutableArrayValueForKey:有序一对多关系成员 NSArray
- mutableSetValueForKey:无序一对多关系成员 NSSet

键值验证(Key-Value Validation)

KVC提供属性值确认的API,它可以用来检查set的值是否正确、为不正确的值做一个替换值或者拒绝设置新值并返回错误原因
调用核查方法:
- validateValue:forKey:error:,默认实现会搜索 validate<Key>:error:格式的核查方法,找到则调用,未找到默认返回YES

集合操作:

集合操作通过对valueForKeyPath:传递参数来使用,一定要用在集合(如:array)上,否则产生运行时刻错误。其格式如下:
Left keypath部分:需要操作对象路径。
Collectionoperator部分:通过@符号确定使用的集合操作。
Rightkey path部分:需要进行集合操作的属性。

  • 数据操作
    @avg:平均值
    @count:总数
    @max:最大
    @min:最小
    @sum:总数
    确保操作的属性为数字类型,否则运行时刻错误。
  1. 对象操作
    针对数组的情况
    @distinctUnionOfObjects:返回指定属性去重后的值的数组
    @unionOfObjects:返回指定属性的值的数组,不去重
    属性的值不能为空,否则产生异常。
  2. 数组操作
    针对数组的数组情况
    @distinctUnionOfArrays:返回指定属性去重后的值的数组
    @unionOfArrays:返回指定属性的值的数组,不去重
    @distinctUnionOfSets:同上,只是返回值为NSSet

KVC键值查找(搜索单值成员)

- setValue:forKey:搜索方式

1、首先搜索setKey:方法。(key指成员变量名,首字母大写)
2、上面的setter方法没找到,如果类方法accessInstanceVariablesDirectly返回YES。那么按 _key,_isKey,key,iskey的顺序搜索成员名。(NSKeyValueCodingCatogery中实现的类方法,默认实现为返回YES)
3、如果没有找到成员变量,调用setValue:forUnderfinedKey:

- valueForKey:的搜索方式

1、首先按getKey,key,isKey的顺序查找getter方法,找到直接调用。如果是BOOL、int等内建值类型,会做NSNumber的转换
2、上面的getter没找到,查找countOfKey、objectInKeyAtindex、KeyAtindexes格式的方法。如果countOfKey和另外两个方法中的一个找到,那么就会返回一个可以响应NSArray所有方法的代理集合的NSArray消息方法
3、还没找到,查找countOfKey、enumeratorOfKey、memberOfKey格式的方法。如果这三个方法都找到,那么就返回一个可以响应NSSet所有方法的代理集合
4、还是没找到,如果类方法accessInstanceVariablesDirectly返回YES。那么按 _key,_isKey,key,iskey的顺序搜索成员名
5、再没找到,调用valueForUndefinedKey

KVC实现分析:

KVC运用了isa-swizzing技术。isa-swizzing就是类型混合指针机制。KVC通过isa-swizzing实现其内部查找定位。isa指针(is kind of 的意思)指向维护分发表的对象的类,该分发表实际上包含了指向实现类中的方法的指针和其他数据。
例如:
[site setValue:@"sitename" forKey:@"name"];
//会被编译器处理成

IMP method = objc_msg_loopup(site->isa,sel);
method(site,sel,@"sitename",@"name");```
每个类都有一张方法表,是一个hash表,值是还书指针IMP,SEL的名称就是查表时所用的键。

SEL数据类型:查找方法表时所用的键。定义成char*,实质上可以理解成int值。
IMP数据类型:他其实就是一个编译器内部实现时候的函数指针。当Objective-C编译器去处理实现一个方法的时候,就会指向一个IMP对象,这个对象是C语言表述的类型

### KVC的内部机制:

一个对象在调用setValue的时候进行了如下操作:
1. 根据方法名找到运行方法的时候需要的环境参数
2. 他会从自己的isa指针结合环境参数,找到具体的方法实现接口。
3. 再直接查找得来的具体的实现方法


## KVO模式

### KVO概念:键值观察Key-Value-Observer就是观察者模式
观察者模式的定义:一个目标对象管理所有依赖于它的观察者对象,并在它自身的状态改变时主动通知观察者对象。这个主动通知通常是通过调用各观察者对象所提供的接口方法来实现的。观察者模式较完美地将目标对象与观察者对象解耦

### KVO实现步骤:
1. 注册监听

//keyPath就是要观察的属性值
//options给你观察键值变化的选择
//context方便传输你需要的数据
-(void)addObserver:(NSObject *)anObserver
forKeyPath:(NSString *)keyPath
options:(NSKeyValueObservingOptions)options
context:(void *)context

2. 实现监听
  ```//change里存储了一些变化的数据,比如变化前的数据,变化后的数据;
  //如果注册时context不为空,这里context就能接收到
  -(void)observeValueForKeyPath:(NSString \*)keyPath
                     ofObject:(id)object
                       change:(NSDictionary \*)change
                      context:(void \*)context
  1. 移除监听

使用观察者模式需要被观察者的配合,当被观察者的状态发生变化的时候通过事先定义好的接口(协议)通知观察者。在KVO的使用中我们并不需要向被观察者添加额外的代码,就能在被观察的属性变化的时候得到通知

KVO实现步骤分析:

  1. 当类A的对象第一次被观察的时候,系统会在运行期动态创建类A的派生类。我们称为B。
  2. 在派生类B中重写类A的setter方法,B类在被重写的setter方法中实现通知机制。
  3. 类B重写会 class方法,将自己伪装成类A。类B还会重写dealloc方法释放资源。
  4. 系统将所有指向类A对象的isa指针指向类B的对象。

KVO同KVC一样,通过 isa-swizzling 技术来实现。当观察者被注册为一个对象的属性的观察对象的isa指针被修改,指向一个中间类,而不是在真实的类。其结果是,isa指针的值并不一定反映实例的实际类。

所以不能依靠isa指针来确定对象是否是一个类的成员。应该使用class方法来确定对象实例的类

使用KVO的几种方法:

  • 使用了KVC
    使用了KVC,如果有访问器方法,则运行时会在访问器方法中调用will/didChangeValueForKey:方法;
    没用访问器方法,运行时会在setValue:forKey方法中调用will/didChangeValueForKey:方法。
  • 有访问器方法
    运行时会重写访问器方法调用will/didChangeValueForKey:方法。
    因此,直接调用访问器方法改变属性值时,KVO也能监听到。
  • 显示调用will/didChangeValueForKey:方法。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容