irpas技术客

iOS-@property的基础知识_Jneth_ios @property

未知 1481

基本概念

@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 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); 对于可变对象类型,如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就会报错了。

除了property中定义外,我们在使用copy方法和mutableCopy方法来复制一个对象,两者的区别在于copy的返回值仍未不可变对象,mutableCopy的返回值为可变对象 typecopymutableCopyNS*浅拷贝,只拷贝指针,地址相同单层深拷贝,拷贝内容,地址不同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"的执行机制

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


1.本站遵循行业规范,任何转载的稿件都会明确标注作者和来源;2.本站的原创文章,会注明原创字样,如未注明都非原创,如有侵权请联系删除!;3.作者投稿可能会经我们编辑修改或补充;4.本站不提供任何储存功能只提供收集或者投稿人的网盘链接。

标签: #iOS #property #_ #copy #Weak #Strong