1.KVO-KVC
KVC的底層實現?
當一個對象調用setValue方法時,方法內部會做以下操作:
①檢查是否存在相應key的set方法,如果存在,就調用set方法
②如果set方法不存在,就會查找與key相同名稱并且?guī)聞澗€的成員屬性,如果有,則直接給成員屬性賦值
③如果沒找到_key,就會查找相同名稱的屬性key,吐過有就直接賦值
④如果還沒找到,則調用valueForUndefinedKey:和setValue:forUndefinedKey:方法
這些方法的默認山西愛你都是拋出異常,我們可以根據需要重寫它們
KVO的底層實現?
- KVO基于runtime機制實現
- 使用了isa混寫(isa-swizzling),當一個對象(假設是person對象,person的類是MYPerson)的屬性值(假設person的age)發(fā)生改變時,系統(tǒng)會自動生成一個派生類,繼承自MYPerson:NSKVONotifying_MYPerson,在這個類的setAge方法里面,調用[super setAge:age] [self willChangeValueForKey:@"age"]和 [self didChangeValueForKey:@"age"],而這兩個方法內部主動調用監(jiān)聽內部的-(void)observeValueForKeyPath這個方法
- 想要看到NSKVONotifying_MYPerson很簡單,在self.person.age = 20;這里打斷點,在調試區(qū)域就能看到_person->NSObject->isa=(Class)NSKVONotifying_MYPerson.同時我們在 self.person = [[MYPerson alloc]init];后面打斷點,看到_person->NSObject->isa=(Class)MYPerson,由此可見,在添加監(jiān)聽者之后,person類型已經由MYPerson被改變成NSKVONotifying_MYPerson
2.什么是KVO和KVC
KVC:鍵值編碼 使用字符串直接訪問對象的屬性
KVO:鍵值觀察機制,他提供了觀察某一屬性變化的方法
3.KVO的缺陷
KVO是一個對象能夠觀察另外一個對象的屬性的值,并且能夠發(fā)現值的變化。前面兩種模式更適合一個controller與任何其他的對象進行通信,而KVO更加適合任何類型的偵聽另外一個任意對象的改變(這里也可以是controller,但一般不是controller。這是一個對象與另外一個對象保持同步的一種做法,即當另外一種對象的狀態(tài)發(fā)生改變時,觀察對象馬上做出反應,而不會用來對方法或者動作做出反應。
優(yōu)點:
1.能夠提供一種簡單的方法實現兩個對象的同步。(例如:model和view之間同步);
2.能夠對非我們創(chuàng)建的對象,即內部對象的狀態(tài)作出響應,而且不需要改變內部對象(SDK對象)的實現;
3.能夠提供觀察的屬性的最新值以及先前值
4.用key paths來觀察屬性,因此也可以觀察嵌套對象
5.完成了對觀察對象的抽象,因為不需要額外的代碼來允許觀察值能夠被觀察
缺點:
1.我們觀察的屬性必須使用strings來定義。因此在編譯器不會出現警告以及檢查
2.對屬性重構將導致我們的觀察代碼不再可用
3.復雜的“IF”語句要求對象正在觀察多個值,這是因為所有的觀察代碼通過通過一個方法來指向
4.當釋放觀察者時需要移除觀察者
Block
1.block:只有普通局部變量是傳值,其他情況都是地址
2.block內存管理
無論在當前環(huán)境是ARC還是MRC,只要blcok沒有訪問外部變量,block始終在全局區(qū)
1.MRC:
- block如果訪問外部變量,block在棧里
- 不能對block使用retain,否則不能保存在堆里
- 只有使用copy,才能放到堆里
2.ARC:
- block如果訪問外部變量,block在堆里
- block可以使用copy和strong,并且block是一個對象
block的循環(huán)引用
- 如果要在block中直接使用外部強指針發(fā)生錯,使用一下代碼在block外部實現可以解決
__weak typeof(self) weakSelf = self;
- 但是如果在block內部使用延時操作還是用弱指針的話會取不到該弱指針,需要在blcok內部再將弱指針強引用一下
__strong typeof(self) strongSelf = weakSelf;
- 如果需要在block內部改變外部變量的話,需要在用__block修飾外部變量
block中的weakself,是任何時候都需要加的么?
不是任何時候都需要添加的,不過任何時候都添加似乎總是好的。只要出現self->block-self.property/self->_ivar這樣的結構鏈時,才會出現循環(huán)引用問題。好好分析一下就可以判斷出是否會有循環(huán)引用問題。