KVO-基本使用方法-底层原理探究-自定义KVO-对容器类的监听
书读百变,其义自见!
将KVO形式以代码实现呈现,通俗易懂,更容易掌握 :GitHub -链接如果失效请自动搜索:https://github.com/henusjj/KVO_base
代码中有详细的注释
一、KVO-常用方法
//注册
- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context; //监听方法
- (void)observeValueForKeyPath:(nullable NSString *)keyPath ofObject:(nullable id)object change:(nullable NSDictionary<NSKeyValueChangeKey, id> *)change context:(nullable void *)context; //移除
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(nullable void *)context API_AVAILABLE(macos(10.7), ios(5.0), watchos(2.0), tvos(9.0));
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath; //监听模式(手动,自动),默认是自动Yes
+(BOOL)automaticallyNotifiesObserversForKey:(NSString *)key; //属性的依赖,返回监听属性类的集合 +(NSSet<NSString *> *)keyPathsForValuesAffectingValueForKey:(NSString *)key;
二、KVO-基本使用
KVO监听属性值变化,从而做业务逻辑处理,监听属性变化,我们需要实现三步走
1.注册监听对象
2.实现监听方法
3.移除监听对象,避免crash
//
// ViewController.m
// KVO-基本用法
//
// Created by GuoYanjun on 2019/1/9.
// Copyright © 2019年 shiyujin. All rights reserved.
// #import "ViewController.h"
#import "Person.h"
@interface ViewController ()
@property(nonatomic,strong)Person *p;
@end @implementation ViewController - (void)viewDidLoad {
[super viewDidLoad];
_p=[[Person alloc]init];
// 注册
[_p addObserver:self forKeyPath:NSStringFromSelector(@selector(name)) options:NSKeyValueObservingOptionNew context:nil]; } //监听方法
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
NSLog(@"%@",change);
}
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{ // 自动模式
static int a =;
_p.name = [NSString stringWithFormat:@"%d",a++]; // 手动
// [_p willChangeValueForKey:@"name"];
// _p.name = [NSString stringWithFormat:@"%d",a++];
// [_p didChangeValueForKey:@"name"];
// }
-(void)dealloc{
[_p removeObserver:self forKeyPath:@"name"];
}
@end
三、底层原理探究
这里我总结三个地方
1.创建一个子类,名字是:NSKVONotifying_Person ,person是本项目中的类
这里为什么是子类不是分类呢,这里说明以下,如果使用分类 ,他会覆盖set方法,导致原set方法中的逻辑处理失效
2.重写了set方法
这里的重写不是重写父类的的set,而是重写子类的
3.外界改变isa指针
此处可以在注册方法打一个断点,观察其isa指针的变化
self->_p->isa:Person
改变为
self->_p->isa:
NSKVONotifying_Person
四、对容器的监听
对于数组,我们添加元素的时候,都是addObject........
但是我们知道,KVO是针对set方法从而监听的,因为,
addObject........是不会响应的,此时,苹果给我提供了
// [_p.arry addObject:[NSString stringWithFormat:@"%d",a++]];//这一步不会触发监听方法因为监听是监听set的方法,addObject不是set方法 //解决方法
NSMutableArray *tenp =[_p mutableArrayValueForKey:@"arry"];
[tenp addObject:[NSString stringWithFormat:@"%d",a++]];
五、自定义KVO
此处需要用到Runtime,KVO文档中,我们会发现,相关方法是在NSobjet的分类,所以
1.创建一个NSObject的分类,定义注册方法
#import <Foundation/Foundation.h> NS_ASSUME_NONNULL_BEGIN @interface NSObject (JK_KVO)
- (void)JK_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
@end NS_ASSUME_NONNULL_END
2.实现该方法
#import "NSObject+JK_KVO.h"
#import <objc/message.h> @implementation NSObject (JK_KVO) - (void)JK_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context{ // 1.创建一个类 -- self.class 就是Person
NSString *oldname = NSStringFromClass(self.class);
NSString *newNem = [@"JKKVO_" stringByAppendingString:oldname];
Class myclass = objc_allocateClassPair(self.class, newNem.UTF8String, );
// 注册类
objc_registerClassPair(myclass); // 2.重写子类set方法 -- 所谓的重写就是给子类添加f这个方法 setName,因为子类没有父类的setName方法!!!
/* class :给那个类添加方法
*sel:方法编号
*imp :方法实现(函数指针)
*type :返回值类型
*/
class_addMethod(myclass, @selector(setName:), (IMP)setName, "v@:@"); // 3.修改isa指针
object_setClass(self, myclass); // 4.将观察保存到当前对象
objc_setAssociatedObject(self, @"observer", observer, OBJC_ASSOCIATION_ASSIGN); } void setName(id self,SEL _cmd,NSString *newName){
NSLog(@"来了--%@",newName);
// 调用父类的setName方法
Class class =[self class];
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:":@""},nil);
} // 改回子类
object_setClass(self, class);
}
@end
3.调用
[_p JK_addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew context:nil];
----!!!!!!!!
OK,结束。代码已经整理完毕。下班
KVO-基本使用方法-底层原理探究-自定义KVO-对容器类的监听的更多相关文章
- jQuery封装自定义事件--valuechange(动态的监听input,textarea)之前值,之后值的变化
jQuery封装自定义事件--valuechange(动态的监听input,textarea)之前值,之后值的变化 js监听输入框值的即时变化 网上有很多关于 onpropertychange.oni ...
- js事件底层原理探究
<!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8&quo ...
- vue自定义组件添加原生事件监听
注:全局或局部注册的组件称为子组件,其中声明的组件名称(如下demo中的child)是一个自定义组件 Demo1-直接给父组件添加事件监听 <!DOCTYPE html> <html ...
- Java基础---Java---IO流-----LineNumberReader方法及原理、自定义一个LineNumberReader、字节流、图片复制、mp3复制、
LineNumberReader 跟综行号的缓冲字符输入流,些类定义了setLineNumber(int)和getLineNumber(int),它们可分别用于设置和获取当前行号 import jav ...
- 虚拟机研究系列-「GC本质底层机制」SafePoint的深入分析和底层原理探究指南
SafePoint前提介绍 在高度优化的现代JVM里,Safepoint有几种不同的用法.GC safepoint是最常见.大家听说得最多的,但还有deoptimization safepoint也很 ...
- JMM和Volatile底层原理分析
JMM和volatile分析 1.JMM:Java Memory Model,java线程内存模型 JMM:它是一个抽象的概念,描述的是线程和内存间的通信,java线程内存模型和CPU缓存模型类似,它 ...
- synchronized底层原理
synchronized底层语义原理 Java 虚拟机中的同步(Synchronization)基于进入和退出管程(Monitor)对象实现. 在 Java 语言中,同步用的最多的地方可能是被 syn ...
- Unity3D热更新之LuaFramework篇[04]--自定义UI监听方法
时隔一个多月我又回来啦! 坚持真的是很难的一件事,其它事情稍忙,就很容易说服自己把写博客的计划给推迟了. 好在终于克服了自己的惰性,今天又开始了. 本篇继续我的Luaframework学习之路. 一. ...
- DownEditTextView【自定义Edittext对Android 软键盘向下的监听】
版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 记录自定义EditText控件实现监听软键盘隐藏事件的功能.基本上和参考资料相同. 效果图 代码分析 自定义EditText子 ...
随机推荐
- LeetCode:19. Remove Nth Node From End of List(Medium)
1. 原题链接 https://leetcode.com/problems/remove-nth-node-from-end-of-list/description/ 2. 题目要求 给出一个链表,请 ...
- 使用postgresql作为cm的数据库时候添加报错
如下图,当postgresql安装成功,建立好数据库scm,rman,amon之后,添加cm对应服务报错hadoopNode2没有相应数据库: No database server found run ...
- THUSC 2018 游记
现在是闭幕式,我坐在西郊宾馆后排,开始写这篇游记. day0 早上从临汾坐火车到北京,12:52左右到了北京. 这次北京的地铁安检没有排成很长的队,但是在买票的时候我惊喜地发现我身上没有零钱--所幸北 ...
- Tapestry 权威讲解-备份
http://blog.csdn.net/mindhawk/article/details/5021371#introduction
- MySQL☞Group By
分组: group by 列名:根据某一列,把数据分成几组,经常对每一组的数据使用聚合函数,按照我的理解,该列有几种不同的值,那么就把该列分成几组,如下图 简单的来说,第二列中有两个不同的值a和b,那 ...
- 12.0 Excel表格读取
Pycharm安装 xlrd 首先在xuexi目录下创建一个ExcelFile文件,让后在ExcelFile下创建一个Excel表格 创建表格时记得把单元格的格式设置为[文本] 我们设置为文本之后,存 ...
- python 基础篇 11 函数进阶----装饰器
11. 前⽅⾼能-装饰器初识本节主要内容:1. 函数名的运⽤, 第⼀类对象2. 闭包3. 装饰器初识 一:函数名的运用: 函数名是一个变量,但他是一个特殊变量,加上括号可以执行函数. ⼆. 闭包什么是 ...
- 1066 Root of AVL Tree (25 分)(平衡二叉树)
就是AVL的模板题了 注意细节 #include<bits/stdc++.h> using namespace std; typedef struct node; typedef node ...
- 简易cmake多文件多目录工程模板
今天心血来潮,想在服务器上试试写libevent的工程是什么感受,那第一步就是学会怎么用cmake建工程,之前也没接触过cmake,然后一上午,比较懵逼,下午看实验室哥们给的一个教程,然后,慢慢理解C ...
- [比赛总结]ACM div3 G 比赛总结
这次题目总体感觉和做阅读理解差不多,英文题目读起来相当费劲. 另外,这次比赛整个队伍中我们三个都突出存在的问题就是,把简单问题复杂化,抓不到事物的本质,因此很容易的就被题目误导. 比如C题,明明想到了 ...