【iOS】KVC的学习

news2025/1/18 6:28:08

【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 {
    @public
    NSString* isAge;
    //NSString* _age;
}
@property NSString* str;
@property NSArray* ary;
@end

NS_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 {
    @public
    NSString* 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;
@end

NS_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 {
    @public
    NSInteger _age;
}
@property NSString* str;
@property NSArray* ary;
@end

NS_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.coloradmin.cn/o/2154259.html

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

相关文章

从零到一,监控网关上网设置教程

要让监控网关成功连接互联网&#xff0c;需要正确配置网络设置。监控网关通常位于本地局域网&#xff08;LAN&#xff09;或广域网&#xff08;WAN&#xff09;中&#xff0c;用于连接摄像头、传感器等监控设备&#xff0c;并通过网络上传数据到远程服务器或云平台。以下是监控…

计算机毕业设计 社区医疗服务系统的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍&#xff1a;✌从事软件开发10年之余&#xff0c;专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精…

【GMNER】Grounded Multimodal Named Entity Recognition on Social Media

Grounded Multimodal Named Entity Recognition on Social Media 动机解决方法特征抽取多模态索引设计索引生成框架EncoderDecoder 实体定位、实体-类型-区域三元组重建 出处&#xff1a;ACL2023 论文链接&#xff1a;https://aclanthology.org/2023.acl-long.508.pdf code链接…

windows安装docker 本地打包代码

参考文章1&#xff1a;https://gitcode.csdn.net/65ea814b1a836825ed792f4a.html 参考文章2&#xff1a; Windows 安装docker&#xff08;详细图解&#xff09;-CSDN博客 一 下载 Docker Desktop 在官网上下载 Docker Desktop&#xff0c;可以从以下链接下载最新版本&#x…

重生之我们在ES顶端相遇第15 章 - ES 的心脏-倒排索引

文章目录 前言为什么叫倒排索引数据结构如何生成如何查询TF、IDF参考文档 前言 上一章&#xff0c;简单介绍了 ES 的节点类型。 本章&#xff0c;我们要介绍 ES 中非常重要的一个概念&#xff1a;倒排索引。 ES 的全文索引就是基于倒排索引实现的。 本章内容建议重点学习&…

基于python的api扫描器系统的设计与实现

&#x1f497;博主介绍&#x1f497;&#xff1a;✌在职Java研发工程师、专注于程序设计、源码分享、技术交流、专注于Java技术领域和毕业设计✌ 温馨提示&#xff1a;文末有 CSDN 平台官方提供的老师 Wechat / QQ 名片 :) Java精品实战案例《700套》 2025最新毕业设计选题推荐…

『功能项目』QFrameWork拾取道具UGUI【69】

本章项目成果展示 我们打开上一篇68QFrameWork扔到地上UGUI的项目&#xff0c; 本章要做的事情是实现当物品在地上时&#xff0c;点击物品将对应物品转移到道具栏中 制作一个提示UI界面 添加Button组件设置为点击即将父物体隐藏 拖拽到文件夹中在场景中删除 创建脚本&#xf…

Postman cURL命令导入导出

导入cURL命令 cURL是一种用于发出HTTP请求的流行命令行工具。在测试Web应用程序或API时&#xff0c;cURL使您能够直接从命令行进行交互&#xff0c;使用API开发人员社区中常见的完善语法。如果在不同的地方有多个cURL命令&#xff0c;可以将它们导入Postman。 ​ 将cURL命令导入…

医院伤员消费点餐限制———未来之窗行业应用跨平台架构

一、点餐上限 医院点餐上限具有以下几方面的意义&#xff1a; 1. 控制成本 - 有助于医院合理规划餐饮预算&#xff0c;避免食物的过度供应造成浪费&#xff0c;从而降低餐饮成本。 2. 保障饮食均衡 - 防止患者或陪护人员过度点餐某一类食物&#xff0c;有利于引导合…

游戏淡入淡出效果

一、制作UIdocument 注&#xff1a;是全黑的&#xff1b;并且Picking Mode设置为Igore 通过调节display中的值&#xff0c;实现淡入淡出效果 二、建立空物体 增加uiDocument 拖入相关的物体 注&#xff1a;层级必须设置为最高&#xff0c;此处为20&#xff0c;这个效果必须遮…

(done) 声音信号处理基础知识(5) (Types of Audio Features for Machine Learning)

参考&#xff1a;https://www.youtube.com/watch?vZZ9u1vUtcIA 声学特征描述了声音&#xff0c;不同特征捕捉声音的不同方面性质 声学特征有助于我们构建智能声学系统 声学特征分类有&#xff1a; 1.抽象等级 2.时域视野 3.音乐的部分 4.信号域 5.机器学习方法 如下图展示…

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…

【Delphi】通过 LiveBindings Designer 链接控件示例

本教程展示了如何使用 LiveBindings Designer 可视化地创建控件之间的 LiveBindings&#xff0c;以便创建只需很少或无需源代码的应用程序。 在本教程中&#xff0c;您将创建一个高清多设备应用程序&#xff0c;该应用程序使用 LiveBindings 绑定多个对象&#xff0c;以更改圆…

[SAP ABAP] 生成表维护视图

SAP由于数据量较大&#xff0c;很多自定义表都需要通过用户自行去维护&#xff0c;一般可以直接在SE16N对数据字典进行维护数据&#xff0c;但不是每个用户都有其操作权限&#xff0c;而且直接在数据字典上操作数据有很高的风险&#xff0c;因此SAP提供了表维护视图生成器&…

算法学习2

学习目录 一.插入排序 一.插入排序 从数组的第一个元素开始&#xff0c;当前元素与其前一个元素进行比较&#xff1b; 大于&#xff08;或小于时&#xff09;将其进行交换&#xff0c;即当前元素替换到前一位&#xff1b; 再将该元素与替换后位置的前一个元素进行交换&#xf…

【全网最全】2024年华为杯研赛A题保奖思路+matlab/py代码+成品论文等(后续会更新完整

您的点赞收藏是我继续更新的最大动力&#xff01; 一定要点击如下卡片链接&#xff0c;那是获取资料的入口&#xff01; 点击链接加入【2024华为杯研赛资料汇总】&#xff1a;https://qm.qq.com/q/goQLLNwfgQhttps://qm.qq.com/q/goQLLNwfgQ A 风电场有功功率优化分配思路 这是…

分页插件、代码生成器

01-分页插件、代码生成器 分页插件使用 首先在pom.xml文件中导入依赖 然后再mybatis-config.xml文件中写入插件 在测试类中写入方法 在mybatis.xml文件中设置plugins标签里的属性helperDialectkeyi自动检查当前数据库用的什么,不用设置也行,默认就设置了 分页插件里面属性详解…

XXL-JOB分片概念讲解

3. 分片功能讲解 3.1 案例需求&#xff1a; 1.我们现在实现这样的需求&#xff0c;在指定节假日&#xff0c;需要给平台的所有用户去发送祝福的短信 3.2.编码实现&#xff1a; a.初始化数据 1.在数据库中导入xxl_job_demo.sql数据 b.集成Druid&MyBatis 1.添加依赖 &…

VisualPromptGFSS

COCO-20 i ^i i太大&#xff0c;不建议复现

利士策分享,华为三折叠手机:重塑未来科技生活的里程碑

利士策分享&#xff0c;华为三折叠手机&#xff1a;重塑未来科技生活的里程碑 在这个日新月异的科技时代&#xff0c;华为再次以惊人的创新力&#xff0c;引领我们迈向智能设备的全新纪元——华为三折叠手机&#xff0c; 不仅是技术的飞跃&#xff0c;更是对未来生活方式的一次…