1.KVO-KVC
KVC的底層實現(xiàn)佑淀?
當一個對象調(diào)用setValue方法時柠新,方法內(nèi)部會做以下操作:
①檢查是否存在相應key的set方法窗价,如果存在冈涧,就調(diào)用set方法
②如果set方法不存在栖袋,就會查找與key相同名稱并且?guī)聞澗€的成員屬性拍顷,如果有,則直接給成員屬性賦值
③如果沒找到_key塘幅,就會查找相同名稱的屬性key,吐過有就直接賦值
④如果還沒找到昔案,則調(diào)用valueForUndefinedKey:和setValue:forUndefinedKey:方法
這些方法的默認山西愛你都是拋出異常,我們可以根據(jù)需要重寫它們
KVO的底層實現(xiàn)电媳?
- KVO基于runtime機制實現(xiàn)
- 使用了isa混寫(isa-swizzling),當一個對象(假設是person對象踏揣,person的類是MYPerson)的屬性值(假設person的age)發(fā)生改變時,系統(tǒng)會自動生成一個派生類匾乓,繼承自MYPerson:NSKVONotifying_MYPerson,在這個類的setAge方法里面捞稿,調(diào)用[super setAge:age] [self willChangeValueForKey:@"age"]和 [self didChangeValueForKey:@"age"],而這兩個方法內(nèi)部主動調(diào)用監(jiān)聽內(nèi)部的-(void)observeValueForKeyPath這個方法
- 想要看到NSKVONotifying_MYPerson很簡單,在self.person.age = 20;這里打斷點拼缝,在調(diào)試區(qū)域就能看到_person->NSObject->isa=(Class)NSKVONotifying_MYPerson.同時我們在 self.person = [[MYPerson alloc]init];后面打斷點娱局,看到_person->NSObject->isa=(Class)MYPerson,由此可見,在添加監(jiān)聽者之后咧七,person類型已經(jīng)由MYPerson被改變成NSKVONotifying_MYPerson
2.什么是KVO和KVC
KVC:鍵值編碼 使用字符串直接訪問對象的屬性
KVO:鍵值觀察機制衰齐,他提供了觀察某一屬性變化的方法
3.KVO的缺陷
KVO是一個對象能夠觀察另外一個對象的屬性的值,并且能夠發(fā)現(xiàn)值的變化猪叙。前面兩種模式更適合一個controller與任何其他的對象進行通信娇斩,而KVO更加適合任何類型的偵聽另外一個任意對象的改變(這里也可以是controller,但一般不是controller仁卷。這是一個對象與另外一個對象保持同步的一種做法穴翩,即當另外一種對象的狀態(tài)發(fā)生改變時犬第,觀察對象馬上做出反應,而不會用來對方法或者動作做出反應芒帕。
優(yōu)點:
1.能夠提供一種簡單的方法實現(xiàn)兩個對象的同步歉嗓。(例如:model和view之間同步);
2.能夠?qū)Ψ俏覀儎?chuàng)建的對象背蟆,即內(nèi)部對象的狀態(tài)作出響應鉴分,而且不需要改變內(nèi)部對象(SDK對象)的實現(xiàn);
3.能夠提供觀察的屬性的最新值以及先前值
4.用key paths來觀察屬性带膀,因此也可以觀察嵌套對象
5.完成了對觀察對象的抽象志珍,因為不需要額外的代碼來允許觀察值能夠被觀察
缺點:
1.我們觀察的屬性必須使用strings來定義。因此在編譯器不會出現(xiàn)警告以及檢查
2.對屬性重構(gòu)將導致我們的觀察代碼不再可用
3.復雜的“IF”語句要求對象正在觀察多個值垛叨,這是因為所有的觀察代碼通過通過一個方法來指向
4.當釋放觀察者時需要移除觀察者
Block
1.block:只有普通局部變量是傳值伦糯,其他情況都是地址
2.block內(nèi)存管理
無論在當前環(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外部實現(xiàn)可以解決
__weak typeof(self) weakSelf = self;
- 但是如果在block內(nèi)部使用延時操作還是用弱指針的話會取不到該弱指針,需要在blcok內(nèi)部再將弱指針強引用一下
__strong typeof(self) strongSelf = weakSelf;
- 如果需要在block內(nèi)部改變外部變量的話佩谷,需要在用__block修飾外部變量
block中的weakself,是任何時候都需要加的么旁壮?
不是任何時候都需要添加的,不過任何時候都添加似乎總是好的谐檀。只要出現(xiàn)self->block-self.property/self->_ivar這樣的結(jié)構(gòu)鏈時抡谐,才會出現(xiàn)循環(huán)引用問題。好好分析一下就可以判斷出是否會有循環(huán)引用問題稚补。