「iOS」——KVC

devtools/2024/10/19 11:18:20/

iOS学习

  • 前言
  • KVC模式
    • KVC设值
    • KVC取值
    • KVC使用keyPath
    • KVC处理异常
      • 处理不存在的key
      • 处理nil异常
    • KVC处理字典
    • KVC高阶消息传递
  • 总结

前言

对KVC模式的简单学习和总结。


KVC模式

KVC(Key-Value Coding,键值编码)是一种通过字符串来访问对象属性的机制,允许开发者通过Key名直接访问对象的属性,或者给对象的属性赋值。而不需要调用明确的存取方法。可以在运行时动态地访问和修改对象的属性。而不是在编译时确定。

以下是KVC的常用方法:

- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;//通过keyPath设置值
- (void)setValue:(id)value forKey:(NSString *)key;//通过key设置值
- (id)valueForKeyPath:(NSString *)keyPath;//通过keyPath获取值
- (id)valueForKey:(NSString *)key;//通过key获取值

KVC设值

我们通过- (void)setValue:(id)value forKey:(NSString *)key;方法来为KVC设值,下面给出代码演示:

#import <Foundation/Foundation.h>@interface AUser : NSObject@property (nonatomic, copy) NSString *str1;
@property (nonatomic, copy) NSString *str2;@end#import <Foundation/Foundation.h>
#import "AUser.h"
#import "AUser1.h"int main(int argc, const char * argv[]) {@autoreleasepool {AUser *user = [[AUser alloc] init];[user setValue:@"Astr1" forKey:@"str1"];[user setValue:@"Astr2" forKey:@"str2"];NSLog(@"str1:%@",[user valueForKey:@"str1"]);NSLog(@"str2:%@",[user valueForKey:@"str2"]);}return 0;
}

运行结果为:
请添加图片描述
那么KVC设置的逻辑原理是什么呢?

请添加图片描述

如上图所示:

  1. 首先会按照setKey、_setKey的顺序查找方法,如找到方法,则直接调用方法并赋值;
  2. 未找到方法,则调用+ (BOOL)accessInstanceVariablesDirectly(是否可以直接访问成员变量,默认返回YES);
  3. 若accessInstanceVariablesDirectly方法返回YES,则按照_key、_isKey、key、isKey的顺序查找成员变量,找到直接赋值,找不到则抛出NSUnknowKeyExpection异常;
  4. 若accessInstanceVariablesDirectly方法返回NO,那么就会调用setValue:forUndefinedKey:并抛出NSUnknowKeyExpection异常;

我们来验证一些处理逻辑:

#import <Foundation/Foundation.h>@interface AUser1 : NSObject{@packageNSString *name;NSString *_name;
}
@end#import <Foundation/Foundation.h>
#import "AUser.h"
#import "AUser1.h"int main(int argc, const char * argv[]) {@autoreleasepool {AUser1 *aUser = [[AUser1 alloc] init];[aUser setValue:@"strName1" forKey:@"_name"];[aUser setValue:@"strName2" forKey:@"name"];NSLog(@"name = %@", aUser->name);NSLog(@"_name = %@", aUser->_name);}return 0;
}

请添加图片描述

KVC取值

我们通过- (id)valueForKey:(NSString *)key; 方法来获取值。
下面来探究取值顺序:

#import "AUser.h"@implementation AUser-(int) getAge{return 9999;
}-(int) age
{return 999;
}-(int) isAge
{return 99;
}
@end#import <Foundation/Foundation.h>
#import "AUser.h"
#import "AUser1.h"int main(int argc, const char * argv[]) {@autoreleasepool {AUser *user = [[AUser alloc] init];[user setValue:@"9" forKey:@"age"];NSLog(@"age:%@",[user valueForKey:@"age"]);}return 0;
}

运行后:

请添加图片描述
注释掉getAge后:

请添加图片描述
注释掉age方法后:
请添加图片描述
再注释掉isAge方法后:
请添加图片描述

原理:

  1. 首先会按照getKey、key、isKey、_key的顺序查找方法,找到直接调用取值
  2. 若未找到,则查看+ (BOOL)accessInstanceVariablesDirectly的返回值,若返回NO,则直接抛出NSUnknowKeyExpection异常;
  3. 若返回的YES,则按照_ key、_isKey、key、isKey的顺序查找成员变量,找到则取值;
  4. 找不到则调用valueForUndefinedKey:抛出NSUnknowKeyExpection异常;

请添加图片描述

KVC使用keyPath

面对复杂的嵌套属性进行初始化赋值时,如果使用key一层层赋值十分麻烦。我们可以采用keyPath来访问对象的嵌套属性。
keyPath的两个方法:

- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;//通过keyPath设置值
- (id)valueForKeyPath:(NSString *)keyPath;//通过keyPath获取值

下面我们来以代码举例:
我们先创建一个BUser:

#import <Foundation/Foundation.h>@interface BUser : NSObject
@property (nonatomic, copy) NSString *strb1;
@property (nonatomic, copy) NSString *strb2;
@end

我们再创建一个AUser嵌套一个BUser:

#import <Foundation/Foundation.h>
#import "BUser.h"@interface AUser : NSObject
{BUser *bUser;
}
@end

此时,AUser对象中,含有一个嵌套属性,我们可以使用keyPath进行为嵌套对象赋值并且取值:

#import <Foundation/Foundation.h>
#import "AUser.h"
#import "BUser.h"int main(int argc, const char * argv[]) {@autoreleasepool {AUser *aUser = [[AUser alloc] init];BUser *bUser = [[BUser alloc] init];[aUser setValue:bUser forKey:@"bUser"];[aUser setValue:@"b1" forKeyPath:@"bUser.strb1"];[aUser setValue:@"b2" forKeyPath:@"bUser.strb2"];NSLog(@"b1:%@",[aUser valueForKeyPath:@"bUser.strb1"]);NSLog(@"b1:%@",[aUser valueForKeyPath:@"bUser.strb2"]);}return 0;
}

打印结果:
请添加图片描述

KVC处理异常

处理不存在的key

在上面我们说过KVC设置的顺序。如果最后没有找到相应的成员变量,则会调用setValue:forUndefinedKey:并抛出NSUnknowKeyExpection异常来结束程序。
我们只需要重写- (void) setValue: (id)value forUndefinedKey:(nonnull NSString *)key 这个方法,则不会产生Crash。


#import "AUser1.h"
@implementation AUser1{int age;
}- (void) setValue: (id)value forUndefinedKey:(nonnull NSString *)key {NSLog(@"重写了setValue:value forUndefinedKey方法");
}
@end#import <Foundation/Foundation.h>
#import "AUser1.h"int main(int argc, const char * argv[]) {@autoreleasepool {AUser1 *aUser = [[AUser1 alloc] init];//处理不存在的key[aUser setValue:@"strName1" forKey:@"1"];}return 0;
}

运行结果:
请添加图片描述

处理nil异常

我们可以将nil赋值给字符串类型,但是不能复制给int或NSInteger类型。如果需要为对象赋nil时,则需要自己处理一下nil异常的部分,例如给int类型赋nil的情况。
当我们给int类型赋值nil时,就会出现异常,会执行-(void)setNilValueForKey:(NSString *)key这个方法,使程序崩溃,所以我们通常需要重写这个方法来处理nil异常。

#import "AUser1.h"@implementation AUser1{int age;
}-(void) setNilValueForKey:(NSString *)key
{if ([key isEqualToString:@"age"]) {age = 0;} else {[super setNilValueForKey:key];}
}
@end#import <Foundation/Foundation.h>
#import "AUser1.h"int main(int argc, const char * argv[]) {@autoreleasepool {AUser1 *aUser = [[AUser1 alloc] init];[aUser setValue:nil forKey:@"age"];NSLog(@"age = %@", [aUser valueForKey:@"age"]);}return 0;
}

运行结果:

请添加图片描述

KVC处理字典

我们可以通过字典进行批量的设值取值操作。

  • setValuesForKeysWithDictionary: 方法用于将字典中的值赋给对象的属性,
  • dictionaryWithValuesForKeys: 方法则用于根据属性键数组获取对象的属性值并返回对应的字典。
#import <Foundation/Foundation.h>@interface Person : NSObject
@property (nonatomic, copy) NSString *name;
@property (nonatomic, copy) NSString *sex;
@property (nonatomic, assign) NSInteger age;@end#import <Foundation/Foundation.h>
#import "Person.h"int main(int argc, const char * argv[]) {@autoreleasepool {Person* person = [[Person alloc] init];//使用KVC批量存值[person setValue:@"Kobe" forKey:@"name"];[person setValue:@"man" forKey:@"sex"];[person setValue:@"24" forKey:@"age"];NSDictionary* firstDictionary = [person dictionaryWithValuesForKeys:@[@"name", @"sex", @"age"]];NSLog(@"dictonary = %@", firstDictionary);//使用KVC批量赋值NSDictionary* secondDictionary = @{@"name":@"瑞娜", @"age":@2, @"sex": @"woman"};Person* secondPerson = [[Person alloc] init];[secondPerson setValuesForKeysWithDictionary:secondDictionary];NSLog(@"name  = %@, age = %ld, sex = %@", secondPerson.name, secondPerson.age, secondPerson.sex);}return 0;
}

KVC高阶消息传递

通俗来讲就是让数组中的每一个元素都执行某个方法,并把结果返回到新的数组中。这里我们实现将数组中的每个首字母大写,并且返回长度。

#import <Foundation/Foundation.h>int main(int argc, const char * argv[]) {@autoreleasepool {NSArray* arrStr = @[@"reyna",@"jett",@"Neon"];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);}}return 0;
}

运行结果:

请添加图片描述


总结

KVC相比于setter和getter方法,虽然在性能上差一点,但是在编码上更加的灵活,简洁,可以批量操作并且可以在运行时动态地访问和操作对象的属性。


http://www.ppmy.cn/devtools/119922.html

相关文章

【机器学习(十)】时间序列案例之月销量预测分析—Holt-Winters算法—Sentosa_DSML社区版

文章目录 一、Holt-Winters算法原理(一) 加法模型(二) 乘法模型(三) 阻尼趋势 二、Holt Winters算法优缺点优点缺点 三、Python代码和Sentosa_DSML社区版算法实现对比(一) 数据读入和统计分析(二) 数据预处理(三) 模型训练和模型评估(四) 模型可视化 四、总结 一、Holt-Winters…

Redis实战-优惠券秒杀

Redis实战篇-短信登入 该笔记是来源于黑马程序员的Redis项目课程,为了后续方便复习。将笔记记录在博客之中 实战篇我们要学习一些什么样的内容 1.本期任务 短信登入 使用redis共享session来实现 商户查询缓存 理解缓存击穿&#xff0c;缓存穿透&#xff0c;缓存雪崩等问题 …

爬虫入门 Selenium使用

爬虫入门 & Selenium使用 特别声明&#x1f4e2;&#xff1a;本教程只用于教学&#xff0c;大家在使用爬虫过程中需要遵守相关法律法规&#xff0c;否则后果自负&#xff01;&#xff01;&#xff01; 项目代码&#xff1a;https://github.com/ziyifast/ziyifast-code_inst…

Redis篇(应用案例 - UV统计)(持续更新迭代)

目录 一、HyperLogLog 二、测试百万数据的统计 一、HyperLogLog 首先我们搞懂两个概念&#xff1a; UV&#xff1a;全称Unique Visitor&#xff0c;也叫独立访客量&#xff0c;是指通过互联网访问、浏览这个网页的自然人。 1天内同一个用户多次访问该网站&#xff0c;只记录…

钉钉如何请求webhook发送信息

代码&#xff1a;js const myHeaders new Headers(); myHeaders.append("Content-Type", "application/json");const raw JSON.stringify({"at": {"atMobiles": ["180xxxxxx"],"atUserIds": ["user123&qu…

【Redis 源码】3dict字典数据结构

1 数据结构说明 dictionary数据结构&#xff0c;也称为哈希表&#xff08;hash table&#xff09;。用于存储字典。哈希是一个键值对的集合&#xff0c;每个键都是唯一的并与一个值相关联。字典的设计旨在提供高效的查找、插入和删除操作。 2 核心数据结构 hash 的数据结构定…

【深度学习】深度学习框架有哪些及其优劣势介绍

本文摘要 在深度学习的开发中&#xff0c;有许多流行的深度学习框架可供使用。本文主要介绍其中一些常见的深度学习框架以及其优劣势。 注&#xff1a;个人观点&#xff0c;仅供学习参考。 原文地址&#xff1a;【深度学习】深度学习框架有哪些及其优劣势介绍 TensorFlow 开发…

ubuntu20.04系统安装zookeeper简单教程

Ubuntu系统中安装和配置Zookeeper的完整指南 Apache Zookeeper是一个开源的分布式协调服务&#xff0c;广泛用于分布式应用程序中管理配置、提供命名服务、分布式同步以及组服务等。在本教程中&#xff0c;我们将详细介绍如何在Ubuntu系统中安装Zookeeper&#xff0c;并进行相关…