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子 ...
随机推荐
- 使用localStorage,sessionStorage,cookie等存储
Web 存储 API 提供了 sessionStorage (会话存储) 和 localStorage(本地存储)两个存储对象来对网页的数据进行添加.删除.修改.查询操作. 特点: localStor ...
- angularjs post data
//post json 时收不到数据,目前只找到方法post form形式的key-value值 //关键是设置 headers: { 'Content-Type': 'application/x- ...
- MySQL基础复习
三范式定义 1NF:每个数据项都是最小单元,不可分割,其实就是确定行列之后只能对应一个数据. 2NF:每一个非主属性完全依赖于候选码(属性组的值能唯一的标识一个元组,但是其子集不可以). 3NF: ...
- struts2官方 中文教程 系列十一:使用XML进行表单验证
在本教程中,我们将讨论如何使用Struts 2的XML验证方法来验证表单字段中用户的输入.在前面的教程中,我们讨论了在Action类中使用validate方法验证用户的输入.使用单独的XML验证文件让 ...
- No parser was explicitly specified, so I'm using the best available HTML parser for this system ("html.parser").警告解决方法
在使用BeautifulSoup库时出现该警告,虽然不影响正常运行,但强迫症不能忍啊!! 详细警告信息如下: UserWarning: No parser was explicitly specifi ...
- 如何从“点子”落地到“执行”?—完整解析1个手游传播类mini项目的进化
本文来自网易云社区 作者:林玮园 从点子到落地,是不确定到确定的过程,是从模糊概念到具体现实的实现过程.无论什么点子,在落地变现的过程中都会有很多疑问产生. 首先,不确定点子本身是否成立.点子的背后是 ...
- 【数据结构】 List 简单实现
public class XList<T> : IEnumerable, IEnumerator { #region List 简单实现 /// <summary> /// 存 ...
- 「Haskell 学习」一 环境与大致了解
感谢<Real World Haskell>在网上的免费发布,可以白嫖学Haskell这个久闻大名的函数式编程语言了. 本文运行于openSUSE Tumbleweed下,运行相关命令时留 ...
- 使用Vue-cli 3.x搭建Vue项目
一.Vue-cli 3.x安装 Node 版本要求:Vue CLI 需要 Node.js 8.9 或更高版本 (推荐 8.11.0+) npm install -g @vue/cli 查版本是否正确 ...
- Oracle数据库抽数神器toad
使用了toad,再也不怕抽数成各种 文件格式,以及添加分割的数据文件了.百度搜toad,