【iOS】KVC的学习

【iOS】KVC的学习

文章目录

  • 【iOS】KVC的学习
    • 前言
    • KVC
      • 定义
      • KVC设值
      • KVC取值
      • KVC使用keyPath
      • KVC处理异常
        • 处理nil异常
      • KVC的一些应用
        • 修改动态的设置值
        • 实现高阶的消息传递
    • 小结

前言

笔者简单学习了有关与KVC的相关内容,这里写一篇博客简单介绍一下相关内容。

KVC

定义

KVC(Key-value coding)键值编码,就是指iOS的开发中,可以允许开发者通过Key名直接访问对象的属性,或者给对象的属性赋值。而不需要调用明确的存取方法。这样就可以在运行时动态地访问和修改对象的属性。而不是在编译时确定,这也是iOS开发中的黑魔法之一。很多高级的iOS开发技巧都是基于KVC实现的。

KVC的定义是通过对于NSObject的扩展来实现的,下面是几个有关于KVC最重要的四个方法:

- (nullable id)valueForKey:(NSString *)key;                          //直接通过Key来取值- (void)setValue:(nullable id)value forKey:(NSString *)key;          //通过Key来设值- (nullable id)valueForKeyPath:(NSString *)keyPath;                  //通过KeyPath来取值- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath;  //通过KeyPath来设值

这里我们从KVC设置值开始说起:

KVC设值

首先我们要清楚KVC是怎么实现一个设置值的效果的,这里我们举一个例子:[test setValue:@"1223" forKey:@"str"];这条语句在执行的时候,KVC到底执行了那些代码。

  • 程序优先调用set:属性值方法,代码通过setter方法完成设置。注意,这里的是指成员变量名,首字母大小写要符合KVC的命名规则,下同
  • 如果没有找到setName:方法,KVC机制会检查+ (BOOL)accessInstanceVariablesDirectly方法有没有返回YES,默认该方法会返回YES,如果你重写了该方法让其返回NO的话,那么在这一步KVC会执行setValue:forUndefinedKey:方法,不过一般开发者不会这么做。所以KVC机制会搜索该类里面有没有名为**_str** 的成员变量,无论该变量是在类接口处定义,还是在类实现处定义,也无论用了什么样的访问修饰符,只在存在以_命名的变量,KVC都可以对该成员变量赋值。
  • 如果该类即没有set:方法,也没有_成员变量,KVC机制会搜索isStr的成员变量。

所以简而言之就是这个设置值的函数查找成员变量的顺序就是顺序查找名称类似于 _<key>_is<Key><key>is<Key> 的实例变量。只要能找到一个满足条件的成员变量就会给这个成员变量设置我们的一个值,下面给一段代码来讲解一下。

#import <Foundation/Foundation.h>NS_ASSUME_NONNULL_BEGIN@interface TestObject : NSObject {@publicNSString* isAge;//NSString* _age;
}
@property NSString* str;
@property NSArray* ary;
@endNS_ASSUME_NONNULL_END
#import "TestObject.h"@implementation TestObject
@end#import <Foundation/Foundation.h>
#import "TestObject.h"
int main(int argc, const char * argv[]) {@autoreleasepool {// insert code here...TestObject* test = [[TestObject alloc] init];[test setValue:@[] forKey:@"ary"];[test setValue:@"1223" forKey:@"str"];[test setValue:@"12" forKey:@"age"];NSLog(@"%@", test->isAge);//NSLog(@"%@", [test valueForKey:@"_age"]);}return 0;
}

打印结果:

12
Type: Notice | Timestamp: 2024-09-17 16:31:02.784471+08:00 | Process: KVC | Library: KVC | TID: 0xda0dbb

可以看到上面这段代码虽然我们不是一个名字为age的标题,但是还是能打印出一个结果。

然后我们修改一下成员变量部分

@interface TestObject : NSObject {@publicNSString* isAge;NSString* _age;
}

主函数

int main(int argc, const char * argv[]) {@autoreleasepool {// insert code here...TestObject* test = [[TestObject alloc] init];[test setValue:@[] forKey:@"ary"];[test setValue:@"1223" forKey:@"str"];[test setValue:@"12" forKey:@"age"];NSLog(@"%@", test->isAge);NSLog(@"%@", test->_age);//NSLog(@"%@", [test valueForKey:@"_age"]);NSLog(@"%@", test.str);NSLog(@"%@", test.ary);}return 0;
}

(null)
Type: Notice | Timestamp: 2024-09-17 17:14:26.416628+08:00 | Process: KVC | Library: KVC | TID: 0xda5d69

12
Type: Notice | Timestamp: 2024-09-17 17:14:26.416670+08:00 | Process: KVC | Library: KVC | TID: 0xda5d69

可以看到我们的isAge这个变量打印出了一个(null)而我们下面的age这个变量则打印出来一个12也就是我们想要的值,这里就说明KVC方法的取值是有一个固定的顺序的。也就是我们上面提到的 _<key>_is<Key><key>is<Key>

同时我们这里如果不想让我们的一个代码实现一个KVC的话,我们可以通过设置+ (BOOL)accessInstanceVariablesDirectly这种方法让他永远返回NO,这样就可以让这个类无法使用KVC。我们先重写下面几个方法,主函数仍旧是上面那一个主函数。

+ (BOOL)accessInstanceVariablesDirectly {return NO;
}
- (id)valueForUndefinedKey:(NSString *)key {NSLog(@"出现异常");return nil;
}
- (void)setValue:(id)value forKey:(NSString *)key {NSLog(@"出现异常,无法设置");
}

打印结果:

出现异常,无法设置。
Type: Notice | Timestamp: 2024-09-17 18:47:05.235218+08:00 | Process: KVC | Library: KVC | TID: 0xdaa7bd

出现异常,无法设置
Type: Notice | Timestamp: 2024-09-17 18:47:05.235285+08:00 | Process: KVC | Library: KVC | TID: 0xdaa7bd

出现异常,无法设置
Type: Notice | Timestamp: 2024-09-17 18:47:05.235285+08:00 | Process: KVC | Library: KVC | TID: 0xdaa7bd

(null)
Type: Notice | Timestamp: 2024-09-17 18:47:05.235326+08:00 | Process: KVC | Library: KVC | TID: 0xdaa7bd

(null)
Type: Notice | Timestamp: 2024-09-17 18:47:05.235326+08:00 | Process: KVC | Library: KVC | TID: 0xdaa7bd

(null)
Type: Notice | Timestamp: 2024-09-17 18:47:05.235326+08:00 | Process: KVC | Library: KVC | TID: 0xdaa7bd

(null)
Type: Notice | Timestamp: 2024-09-17 18:47:05.235340+08:00 | Process: KVC | Library: KVC | TID: 0xdaa7bd

重写这个方法后,如果我们的KVC找不到set< Key >这个方法之后就不会继续向下寻找了。

在这里插入图片描述

正如上图所示,KVC的设值大概就是上图的过程。

KVC取值

有关于KVC取值,一般采用

  • 首先按get< Key >,< key>,is < Key>的顺序方法查找getter方法,找到的话会直接调用。如果是BOOL或者Int等值类型, 会将其包装成一个NSNumber对象。
  • 如果还没有找到,再检查类方法+ (BOOL)accessInstanceVariablesDirectly,如果返回YES(默认行为),那么和先前的设值一样,会按_,_is,is的顺序搜索成员变量名,这里不推荐这么做,因为这样直接访问实例变量破坏了封装性,使代码更脆弱。

一般情况按照这三个顺序进行一个查找,笔者这里给出函数。

#import "TestObject.h"@implementation TestObject
-(int)getAge {return 1222;
}
-(int)age {return 120;
}
-(int)isAge {return 10;
}
@end
#import <Foundation/Foundation.h>
#import "TestObject.h"
int main(int argc, const char * argv[]) {@autoreleasepool {// insert code here...TestObject* test = [[TestObject alloc] init];[test setValue:@[] forKey:@"ary"];[test setValue:@"1223" forKey:@"str"];[test setValue:@"12" forKey:@"age"];NSLog(@"%@",[test valueForKey:@"age"]);//NSLog(@"%@", test->_age);//NSLog(@"%@", [test valueForKey:@"_age"]);NSLog(@"%@", test.str);//NSLog(@"%@", test.ary);}return 0;
}

打印结果:

1222
Type: Notice | Timestamp: 2024-09-18 22:58:51.536732+08:00 | Process: KVC | Library: KVC | TID: 0xdf2afb

返回的是我们getAge的方法返回的函数。

这时候我们注释getAge,然后重新运行一下

120
Type: Notice | Timestamp: 2024-09-18 23:00:38.285681+08:00 | Process: KVC | Library: KVC | TID: 0xdf33e6

最后我们在注释age,在运行一下得到

10
Type: Notice | Timestamp: 2024-09-18 23:02:06.668767+08:00 | Process: KVC | Library: KVC | TID: 0xdf3d01

最后我们注释isAge,运行一下可以得到。

12
Type: Notice | Timestamp: 2024-09-18 23:05:09.897443+08:00 | Process: KVC | Library: KVC | TID: 0xdf4c04

KVC使用keyPath

有时候我们的要改变的对象可能是比较复杂的,比如说自定义类或者是其他复杂的数据类型,我们如果使用key来一层一层的监控的话,会非常复杂,这时候就出现了keyPath这个方法来简化代码。

如下面的代码,我们先定义一个新的类。

#import <Foundation/Foundation.h>
#import "TestObject.h"
NS_ASSUME_NONNULL_BEGIN@interface NextText : NSObject
@property TestObject* test;
@endNS_ASSUME_NONNULL_END

上面这个自定义类存储了一个属性是我们之前创建的对象。

int main(int argc, const char * argv[]) {@autoreleasepool {// insert code here...TestObject* test = [[TestObject alloc] init];NextText* test1 = [[NextText alloc] init];[test1 setValue:test forKey:@"test"];[test1 setValue:@"1222" forKeyPath:@"test.str"];NSLog(@"%@", [test1 valueForKeyPath:@"test.str"]);}return 0;
}

打印结果:

1222
Type: Notice | Timestamp: 2024-09-19 18:24:49.998580+08:00 | Process: KVC | Library: KVC | TID: 0xe07e3c

这里可以看到我们在这里也是成功设置了有关keyPath的内容,然后也通过keyPath来实现了获取对应的数值。所以说我们仅仅需要用一个点语法的形式就可以实现一个修改对应路径的一个值。

KVC处理异常

在一般情况下,我们是不允许KVC来给一个对象赋值为nil的,这时候,我们可能需要自己去处理一下nil异常的部分。比方说,我们传入一个nil给NSInteger的对象的时候,就会出现一个异常的问题,会执行-(void)setNilValueForKey:(NSString *)key这个方法,这个方法会产生一个Crash,所以我们通常需要重写这个方法。

处理nil异常
#import <Foundation/Foundation.h>NS_ASSUME_NONNULL_BEGIN@interface TestObject : NSObject {@publicNSInteger _age;
}
@property NSString* str;
@property NSArray* ary;
@endNS_ASSUME_NONNULL_END
#import "TestObject.h"@implementation TestObject
- (void)setNilValueForKey:(NSString *)key {if ([key isEqualToString:@"age"]) {NSLog(@"不能设置成nil状态%@", key);_age = 11;} else {[super setNilValueForKey:key];}
}
@end
#import <Foundation/Foundation.h>
#import "TestObject.h"
#import "NextText.h"
int main(int argc, const char * argv[]) {@autoreleasepool {// insert code here...TestObject* test = [[TestObject alloc] init];[test setValue:nil forKey:@"age"];NSLog(@"%@", [test valueForKey:@"age"]);//NSLog(@"%@", test.ary);}return 0;
}

然后执行代码的打印结果是:

不能设置成nil状态age
Type: Notice | Timestamp: 2024-09-19 19:31:52.434134+08:00 | Process: KVC | Library: KVC | TID: 0xe16087

11
Type: Notice | Timestamp: 2024-09-19 19:31:52.434305+08:00 | Process: KVC | Library: KVC | TID: 0xe16087

Program ended with exit code: 0
Type: stdio

KVC的一些应用

修改动态的设置值

这是KVC最简单的应用。

实现高阶的消息传递

首先,我们要明白对于容器使用一个KVC,并没有对于我们的容器进行一个操作,实际上是将这个方法传递给容器中的每一个元素,然后再重新返回一个容器,所以这里我们实现一些特殊的效果,通过操作一个容器来返回一个符合我们需求的容器。

int main(int argc, const char * argv[]) {@autoreleasepool {// insert code here...NSArray* arrStr = @[@"english",@"franch",@"chinese"];NSArray* arrCapStr = [arrStr valueForKey:@"capitalizedString"];for (NSString* str  in arrCapStr) {NSLog(@"%@",str);}NSArray* arrCapStrLength = [arrStr valueForKeyPath:@"capitalizedString.length"];for (NSNumber* length  in arrCapStrLength) {NSLog(@"%ld",(long)length.integerValue);}//NSLog(@"%@", test.ary);}return 0;
}

打印结果:

English
Type: Notice | Timestamp: 2024-09-19 19:52:26.638075+08:00 | Process: KVC | Library: KVC | TID: 0xe198e4
Franch
Type: Notice | Timestamp: 2024-09-19 19:52:26.638130+08:00 | Process: KVC | Library: KVC | TID: 0xe198e4
Chinese
Type: Notice | Timestamp: 2024-09-19 19:52:26.638150+08:00 | Process: KVC | Library: KVC | TID: 0xe198e4
7
Type: Notice | Timestamp: 2024-09-19 19:52:26.638865+08:00 | Process: KVC | Library: KVC | TID: 0xe198e4
6
Type: Notice | Timestamp: 2024-09-19 19:52:26.638895+08:00 | Process: KVC | Library: KVC | TID: 0xe198e4
7
Type: Notice | Timestamp: 2024-09-19 19:52:26.638912+08:00 | Process: KVC | Library: KVC | TID: 0xe198e4
Program ended with exit code: 0
Type: stdio

可以看到这里我们通过KVC来实现了一个让容器中所有字符串第一个字符大写的容器。

小结

笔者在这里简单介绍了一下有关于KVC的使用,之后还会继续学习相关的内容。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1541872.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

saas收银系统源码

1. 线下门店多样化收银 ①门店有社区小店、也会有大店&#xff0c;甚至还会有夫妻店&#xff0c;同时还要有Windows版和安卓版&#xff0c;需满足不同门店的收银需求。 ②支持Windows收银、安卓收银、无人自助收银、聚合码收银等&#xff0c;支持ai智能称重、收银称重一体机等…

『功能项目』QFrameWorkBug拖拽功能【66】

我们打开上一篇65QFrameWork道具栏物品生成的项目&#xff0c; 本章要做的事情是实现物品的拖拽功能 修改脚本&#xff1a;UISlot.cs 实现接口后编写脚本&#xff1a; using UnityEngine; using UnityEngine.EventSystems; using UnityEngine.UI; namespace QFramework {publi…

Netty+HTML5+Canvas 网络画画板实时在线画画

采用Html5的canvas做前端画画板&#xff0c;发送数据到后端Netty服务&#xff0c;实时转发笔迹数据&#xff0c;在线实时同步画笔轨迹&#xff0c;单击绿色小方块&#xff0c;保存画板的图片 页面&#xff1a; <!-- index.html --><!DOCTYPE html> <html> …

[Linux]:信号(下)

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;Linux学习 贝蒂的主页&#xff1a;Betty’s blog 1. 信号的阻塞 1.1 基本概念 信号被操作系统发送给进程之后&#xff0c;进程…

机器学习05-聚类算法(python)SC(轮廓系数)详解

# 导入必要的库 from sklearn.cluster import KMeans # 导入 KMeans 聚类算法 import matplotlib.pyplot as plt # 导入 matplotlib 用于绘图 from sklearn.datasets import make_blobs # 导入 make_blobs 用于生成模拟数据 from sklearn.metrics import silhouette_score …

react:组件通信

组件通信 父组件向子组件通信 function App() {return (<div><div>这是父组件</div><Child name"这是子组件" /></div>); }// 子组件 function Child(props) {return <div>{props.name}</div>; }props说明 props可以传…

浅谈计算机视觉的学习路径1

计算机视觉&#xff08;Computer Vision, CV&#xff09;是人工智能领域的一个重要分支&#xff0c;它的目标是使计算机能够像人类一样理解和处理图像和视频数据。 面向想要从事该方向的大学生&#xff0c;笔者这里给出以下是关于计算机视觉的学习路径建议&#xff1a; 简要了解…

Linux开发工具(git、gdb/cgdb)--详解

目录 一、Linux 开发工具分布式版本控制软件 git1、背景2、使用 git&#xff08;1&#xff09;预备工作——安装 git&#xff1a;&#xff08;2&#xff09;克隆远程仓库到本地&#xff08;3&#xff09;把需要提交的代码拷贝到本地仓库&#xff08;4&#xff09;提交本地仓库文…

一种新的电子邮件攻击方式:AiTM

新的攻击组利用合作伙伴组织之间的信任关系来绕过多重身份验证。 一种新的攻击方式开始出现&#xff0c;它利用合作伙伴组织之间的信任关系绕过多重身份验证。在一个利用不同组织之间关系的攻击中&#xff0c;攻击者成功地对四家或更多组织进行了商业电子邮件欺诈(BEC)攻击&…

VM-Ubantu中使用vscode头文件报错——解决办法

问题 系统中头文件明明存在但是却报错 解决方法 在报错的文件中点击&#xff0c;shift ctrl p选择Edit Configurations(JSON) 修改文件内容 原文件内容 修改之后的内容 {"configurations": [{"name": "Linux","includePath":…

计算机毕业设计推荐-基于python大数据的个性化图书数据可视化分析

&#x1f496;&#x1f525;作者主页&#xff1a;毕设木哥 精彩专栏推荐订阅&#xff1a;在 下方专栏&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; 实战项目 文章目录 实战项目 一、个性化图书数据可视化分析-项…

828华为云征文 | 云服务器Flexus X实例:开源项目 LangChain 部署,实例测试

目录 一、LangChain 介绍 二、部署 LangChain 2.1 安装 langchain 2.2 安装 langchain_community 2.3 安装 qianfan 三、实例运行 3.1 Chat Models 3.2 LLMs 3.3 Embedding Models 四、总结 本篇文章主要通过 Flexus云服务器X实例 部署开源项目 LangChain&#xff0c…

【每日一题】LeetCode 2374.边积分最高节点(图、哈希表)

【每日一题】LeetCode 2374.边积分最高节点&#xff08;图、哈希表&#xff09; 题目描述 给定一个有向图&#xff0c;图中包含 n 个节点&#xff0c;节点编号从 0 到 n - 1。每个节点都有一个出边&#xff0c;指向图中的另一个节点。图由一个长度为 n 的整数数组 edges 表示…

【Linux学习】基本指令其一

命令行界面 命令行终端是一个用户界面&#xff0c;允许用户通过输入文本命令与计算机系统进行交互。 比如Windows下&#xff0c; 键入winR&#xff0c;然后输入cmd&#xff0c;就可以输入文本指令与操作系统交互了。 Windows有另一个命令行界面Powershell,它的功能比cmd更强大…

江协科技STM32学习- P15 TIM输出比较

&#x1f680;write in front&#x1f680; &#x1f50e;大家好&#xff0c;我是黄桃罐头&#xff0c;希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流 &#x1f381;欢迎各位→点赞&#x1f44d; 收藏⭐️ 留言&#x1f4dd;​…

【开源】LVGL+FreeRTOS 基于STM32F411CEU6的健康助手项目制作

视频演示 【开源】LVGLFreeRTOS 基于STM32F411的智能健康助手小项目 网盘链接在最底下&#xff01;&#xff01;&#xff01;无套路&#xff01;&#xff01;&#xff01;直接分享&#xff01;&#xff01;&#xff01; 硬件介绍 STM32F411CEU6 主控 TFT 1.8inch 显示屏 DTH…

WebGL缓冲区

一、缓冲区对象 缓冲区对象时WebGL系统中的一块内存区域&#xff0c;可以一次性地向缓冲区对象中填充大量的顶点数据&#xff0c;然后将这些数据保存其中&#xff0c;供顶点着色器使用。 类型化数组 这样程序可以预知数组中的类型&#xff0c;提高性能 类型描述Int8Array8位…

数据湖 Data Lake-概述

Data Lake 1. 数据湖的定义 数据湖是一种存储系统&#xff0c;用于集中存储大量的原始数据&#xff0c;可以按数据本来的原始格式进行存储&#xff0c;用户可以在需要时提取和分析这些数据。 A data lake is a centralized repository designed to hold vast volumes of data …

JavaScript高级进阶(三)

DOM-改变HTML 语法与说明 document.write() //改变HTML输出流&#xff0c;整个页面进行重绘。 操作对象.innerHTML新的HTML //改变HTML内容 操作对象.attribute新属性值 //改变HTML属性 对象.style.property新样式 //改变操作样式的属性 注意: document.write(),优先级太高&am…

Th:1.1 建立连接

基础讲解 1.TCP通信流程 基于TCP通信的Socket基本流程: 1.1 Socket 函数返回值&#xff1a;一个文件描述符&#xff1a; 特别的两个队列。 #include <sys/types.h> #include <sys/socket.h> //create an endpoint for communication int socket(int …