首先定义一个测试用的类Test
Test.h
#import <Foundation/Foundation.h>
@interface Test : NSObject
@property (nonatomic, copy, readonly) NSString *testName;
@property (nonatomic, assign, readonly) NSInteger testAge;
- (instancetype)initWithTestName:(NSString *)name testAge:(NSInteger)age;
@end
复制代码
Test.m
#import "Test.h"
@implementation Test
- (instancetype)initWithTestName:(NSString *)name testAge:(NSInteger)age{
self = [super init];
if (self) {
_testName = name;
_testAge = age;
}
return self;
}
@end
复制代码
然后定义一个Test的类对象:
Test *test = [[Test alloc] initWithTestName:@"testName" testAge:22];
NSLog(@"%@",test.testName);
复制代码
如果我们直接调用testName的setter方法,test.testName = @"修改了";会直接报错。提示这个属性是只读的。
如果我们使用KVC呢?
Test *test = [[Test alloc] initWithTestName:@"testName" testAge:22];
NSLog(@"-%@",test.testName);
[test setValue:@"修改了" forKey:@"testName"];
NSLog(@"---%@",test.testName);
复制代码
输出结果:
2017-05-17 14:20:36.404 iOS_readonlyTest[4076:115282] -testName
2017-05-17 14:20:36.404 iOS_readonlyTest[4076:115282] ---修改了
复制代码
我们看到,使用KVC成功修改了声明为readonly的属性。
如果我们不想让 setValue:forKey: 方法改变对象的属性值,那么重写其类方法 + (BOOL)accessInstanceVariablesDirectly 返回 NO (该方法默认返回 YES,即在不存在满足条件的存取方法时,允许直接访问属性对应的实例变量);在搜索实例变量时,会首先检查带下划线的实例变量,然后检查不带下划线的实例变量。
提示:
重写其类方法 + (BOOL)accessInstanceVariablesDirectly 返回 NO的情况下,利用KVC修改声明为readonly的属性的值的时候会崩溃。我们可以重写类的setValue:forKey: 方法,判断key是否是声明为readonly的属性,如果是直接返回。
- (void)setValue:(id)value forKey:(NSString *)key{
if ([key isEqualToString:@"testName"]) {
NSLog(@"这个属性不能被修改");
return;
}
[super setValue:value forKey:key];
}
复制代码
2017-05-17 14:29:35.894 iOS_readonlyTest[4108:118687] -testName
2017-05-17 14:29:35.894 iOS_readonlyTest[4108:118687] 这个属性不能被修改
2017-05-17 14:29:35.895 iOS_readonlyTest[4108:118687] ---testName
复制代码
参考链接:http://www.jianshu.com/p/1ffa6414008e