之前我們已經(jīng)了解過了KVO的底層實現(xiàn)原理,不過呢,在我們開始實現(xiàn)自定義KVO之前再來簡單回顧下KVO的實現(xiàn)原理
1.創(chuàng)建子類
2.重寫一個setter方法(其實是添加一個setter方法)
3.修改isa指針指向新創(chuàng)建的子類
4.調(diào)用父類的setName方法
5.將觀察者保存到當(dāng)前對象
接下來我們開始自定義KVO
我們先創(chuàng)建一個NSObject分類;然后實現(xiàn)對應(yīng)的方法
#import <Foundation/Foundation.h>
NS_ASSUME_NONNULL_BEGIN
@interface NSObject (KVO)
- (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
@end
NS_ASSUME_NONNULL_END
1.創(chuàng)建子類
在.m文件里#import <objc/message.h>
創(chuàng)建完子類記得要注冊這個類,
- (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
//1.創(chuàng)建子類
NSString *oldClassName = NSStringFromClass(self.class);
NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
// 注冊
objc_registerClassPair(myClass);
}
2.重寫一個setter方法(其實是添加一個setter方法)
- (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
//1.創(chuàng)建子類
NSString *oldClassName = NSStringFromClass(self.class);
NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
// 注冊
objc_registerClassPair(myClass);
//2.重新setNanme方法(其實是添加一個setName方法)
class_addMethod(myClass, @selector(setName:), (IMP)setMethod, "v@:@");
}
void setMethod(id self,SEL _cmd,NSString *newName){
NSLog(@"來了%@",newName);
}
3.修改isa指針指向新創(chuàng)建的子類
- (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
//1.創(chuàng)建子類
NSString *oldClassName = NSStringFromClass(self.class);
NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
// 注冊
objc_registerClassPair(myClass);
//2.重新setNanme方法(其實是添加一個setName方法)
class_addMethod(myClass, @selector(setName:), (IMP)setMethod, "v@:@");
//3.修改imp地址
object_setClass(self, myClass);
}
這個時候我們在控制器中注冊這個觀察者( 其中person是需要觀察屬性變化的實體類)
- (void)viewDidLoad {
[super viewDidLoad];
person = [[Person alloc]init];
[person ZXY_addObserver:self forKeyPath:@"name" options:(NSKeyValueObservingOptionNew) context:nil];
}
//點(diǎn)擊屏幕
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
static int a;
person.name = [NSString stringWithFormat:@"%d",a++];
}
然后我們跑起來,點(diǎn)擊屏幕: 控制臺打印
屏幕快照 2018-12-25 下午4.33.43.png
4.調(diào)用父類的setName方法
到目前這個階段,我們已經(jīng)能獲取到修改的值了,接下來我們需要調(diào)用父類的set方法,去修改屬性值,這樣person的name就修改了.
void setName(id self,SEL _cmd,NSString *newName){
NSLog(@"來了%@",newName);
//5. 調(diào)用父類的setName方法
Class class = [self class];//拿到當(dāng)前類型
object_setClass(self, class_getSuperclass(class));// 將當(dāng)前類型設(shè)置為父類類型
objc_msgSend(self, @selector(setName:),newName);//給父類的set方法發(fā)送消息,傳遞修改的值
// 改為子類
object_setClass(self, class);
}
注意:在這一步一定要將當(dāng)前的類型設(shè)置為父類類型,設(shè)置完后也一定要改為子類的類型
5.將觀察者保存到當(dāng)前對象
現(xiàn)在我們需要通知外部方法,已經(jīng)將屬性修改了,并將之傳遞出去,所以我們設(shè)置一個觀察者
- (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
//1.創(chuàng)建子類
NSString *oldClassName = NSStringFromClass(self.class);
NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
// 注冊
objc_registerClassPair(myClass);
//2.重新setNanme方法(其實是添加一個setName方法)
/**參數(shù)
*class 給哪個類添加方法
*sel 方法編號
*imp 方法實現(xiàn)(函數(shù)指針)
*type 返回值類型 ()
*/
class_addMethod(myClass, @selector(setName:), (IMP)setMethod, "v@:@");
//3.修改imp地址
object_setClass(self, myClass);
//4.將觀察者保存到當(dāng)前對象
/**
id object :表示關(guān)聯(lián)者鲸郊,是一個對象,變量名理所當(dāng)然也是object
const void *key :獲取被關(guān)聯(lián)者的索引key
id value :被關(guān)聯(lián)者,這里是一個block
objc_AssociationPolicy policy : 關(guān)聯(lián)時采用的協(xié)議,有assign,retain缴守,copy等協(xié)議,一般使用OBJC_ASSOCIATION_RETAIN_NONATOMIC,這里我們使用Weak協(xié)議,避免循環(huán)引用
*/
objc_setAssociatedObject(self, "observer", observer, OBJC_ASSOCIATION_ASSIGN);//
}
在這里就有可能就有人問了,明明set方法只有一個返回類型,為什么要寫“v@:@”,這里解釋下,V表示的是返回void類型,第二個@表示的是一個id對象,第三個:表示的是方法SEL,最后一個@表示的傳入的變量值.
??注意:以為OC方法會默認(rèn)兩個參數(shù),一個id對象,一個方法SEL
然后我們在set方法中獲取到這個觀察者
void setName(id self,SEL _cmd,NSString *newName){
NSLog(@"來了%@",newName);
//5. 調(diào)用父類的setName方法
Class class = [self class];//拿到當(dāng)前類型
object_setClass(self, class_getSuperclass(class));
objc_msgSend(self, @selector(setName:),newName);
// 觀察者
id observer = objc_getAssociatedObject(self, "observer");
// 改為子類
object_setClass(self, class);
}
給系統(tǒng)的observeValueForKeyPath方法發(fā)送消息
void setName(id self,SEL _cmd,NSString *newName){
NSLog(@"來了%@",newName);
//5. 調(diào)用父類的setName方法
Class class = [self class];//拿到當(dāng)前類型
object_setClass(self, class_getSuperclass(class));
objc_msgSend(self, @selector(setName:),newName);
// 觀察者
id observer = objc_getAssociatedObject(self, "observer");
if (observer) {
objc_msgSend(observer, @selector(observeValueForKeyPath:ofObject:change:context:),@"name",self,@{@"new:":newName,@"kind:":@1},nil);
}
// 改為子類
object_setClass(self, class);
}
其中observeValueForKeyPath方法的參數(shù)說明:
keyPath: 屬性名字
object: 哪個對象
change: 一個字典,它描述對鍵路徑keyPath中的屬性值相對于對象所做的更改
context: 當(dāng)注冊觀察者以接收鍵值觀察通知時提供的值,這里寫為nil就行.
最后
現(xiàn)在我們在外部控制器中打印修改后的值
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
NSLog(@"%@",change);
}
屏幕快照 2018-12-26 下午2.23.06.png
到這里,自定義KVO就簡單實現(xiàn)了,但是目前里面的屬性變量名都是固定,如果能自定義的變化,那就更完美了.降下來,我們就來慢慢的研究他.