iOS-@property的基础知识

基本概念

@property用于定义属性,将会自动生成getter和setter方法,并生成带 “_” 的成员变量,同时属性提供了一些可能会使用的特性来进行声明,包括assign(vs copy),weak,strong,atomic(vs nonatomic),readwrite,readonly等
@property本质:
@property = ivar(实例变量) + getter/setter(存取方法);

编译关键字

  • @synthesize表示为这两个属性自动生成名为“_属性”的底层实例变量,并自动生成相关的getter和setter
  • @dynamic 告诉编译器:属性的 setter 与 getter 方法由用户自己实现,应该也不会有实例变量

@property指示符

1.@property指示符readonly

系统只会生成getter方法不会有setter方法,所以无法对该属性赋值,但可以通过kvc赋值。在后续介绍。

2.@property指示符class

@property (class) NSString *someStringProperty; Objective-C现在支持与Swift类型属性进行交互转换的类属性,这些属性永远不会synthesized,不会生成set和get方法。

3.@property指示符copy
  1. copy修饰的属性会在内存里拷贝一份对象,两个指针指向不同的内存地址。
    一般用来修饰有对应可变类型子类的对象。
    NSString/NSMutableString,NSArray/NSMutableArray,NSDictionary/NSMutableDictionary等。

为确保这些不可变对象因为可变子类对象影响,需要copy一份备份,如果不使用copy修饰,使用strong或assign等修饰则会因为多态导致属性值被修改

Person *p = [[Person alloc] init];
NSMutableString *s = [[NSMutableString alloc]       initWithString:@"Jiaming Chen"];
//将可变字符串赋值给p.name
p.name = s;
//输出的地址和内容均一致
NSLog(@"%p %p %@ %@", p.name, s, p.name, s);
//修改可变字符串s
[s appendString:@" is a good guy"];
//再次输出p.name被影响
NSLog(@"%p %p %@ %@", p.name, s, p.name, s);
  1. 对于可变对象类型,如NSMutableString、NSMutableArray等则在property不可以使用copy修饰(OC属性里没有mutableCopy修饰符),因为Foundation框架提供的这些类都实现了NSCopying协议,使用copy方法返回的都是不可变对象。
Person *p = [[Person alloc] init];
NSMutableString *s = [[NSMutableString alloc] initWithString:@"Jiaming Chen"];
//将可变字符串赋值给p.name
p.name = s;
//输出的地址不一致,内容一致
NSLog(@"%p %p %@ %@", p.name, s, p.name, s);
//修改p.name,此时抛出异常
[p.name appendString:@" is a good guy."];

这里虽然定义了name为MutableString但是在运行时遵循copy协议,p.name的类型会是NSString类型,调用appendString就会报错了。

  1. 除了property中定义外,我们在使用copy方法和mutableCopy方法来复制一个对象,两者的区别在于copy的返回值仍未不可变对象,mutableCopy的返回值为可变对象
typecopymutableCopy
NS*浅拷贝,只拷贝指针,地址相同单层深拷贝,拷贝内容,地址不同
NSMutable*单层深拷贝,拷贝内容,地址不同单层深拷贝,拷贝内容,地址不同

修饰NSString这样的不可变对象的时候使用copy修饰,但其实当给对象赋一个NSString时仍旧只复制了指针而不是拷贝内容

4.@property指示符assign

assign表示对属性只进行简单的赋值操作,不更改所赋的新值的引用计数,也不改变旧值的引用计数。所以问题就出在当给一个变量p.A赋值为另一个变量B时,将B设为nil,而p.A的地址依然可以访问,但是访问内容时就会野指针了。

5.@property指示符weak

weak同assign,但是被weak修饰的属性,该值内容被销毁时,weak修饰的属性会被自动赋值为nil。常用与delegate中防止循环引用。

KVC对属性的操作

KVC(key-Value coding) 键值编码,指iOS开发中,可以允许开发者通过Key名直接访问对象的属性,或者给对象的属性赋值。不需要调用明确的存取方法,这样就可以在运行时动态访问和修改对象的属性,而不是在编译时确定。
KVC的定义都是对NSObject的扩展来实现的,OC中有个NSKeyValueCoding类别名。

setvalue:属性值 forKey:@"key"的执行机制

  1. 程序优先会通过setter方法进行设置
  2. 若没有setter方法(如readonly和class修饰符),则检查+(BOOL)accessInstanceVariablesDirectly方法若返回yes则去查找成员变量_key,之后还可能会查找key和isKey成员变量进行赋值。注意这里如果我们重写了一个类的accessInstanceVariablesDirectly方法并返回NO就可以做到禁用kvc。
  3. 若查找不到则调用setValue:forUndefinedKey:

补充:kvc操作集合@avg、@count、@max、@min、@sum5种

NSArray *arrBooks = @[book1,book2,book3,book4];
NSNumber *sum = [arrBooks valueForKeyPath:@"@sum.price"];

KVO的实现

当一个对象使用了KVO监听,iOS系统会修改这个对象的isa指针,改为指向一个全新的通过Runtime动态创建的子类,子类拥有自己的set方法实现,set方法实现内部会顺序调用willChangeValueForKey方法、原来的setter方法实现、didChangeValueForKey方法,而didChangeValueForKey方法内部又会调用监听器的observeValueForKeyPath:ofObject:change:context:监听方法。

被监听的属性的值被修改时,就会自动触发KVO。如果想要手动触发KVO,则需要我们自己调用willChangeValueForKey和didChangeValueForKey方法即可在不改变属性值的情况下手动触发KVO,并且这两个方法缺一不可。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值