属性的声明和修饰

@property

@property修饰的变量,会通过编译器加入@synthesize自动合成为ivar+getter+setter(property_t),而且会生成成员变量,变量的名字就是_+属性名字,当然,不想用默认的带下划线的,你也可以通过@synthesize test = myTest指定实例变量名。@synthesize test =  _test主要用于重写了setter方法、override父类、dynamic等不会自动合成ivar和存取器的情况时,做一下手动绑定。

在category中使用@property:不会添加ivar,不会有成员变量,只会生成getter/setter的方法声明,所以需要通过get/setAssociatedObject自己实现。

注1:成员变量 = 实例变量(id) + 基本类型(int)。

注2:使用@dynamic可以取消自动合成,自己实现setter/getter方法,没实现将会crash。


weak / strong / copy

weak原理

每一个有弱引用的对象,系统都维护一个表SideTable中的weak_table,记录它所有的弱引用指针地址而不增加引用计数,引用计数为0时找到这张表里所有的弱引用指针,置为nil。

对象对应的sidetable→weak_table→对象的地址作为key→weak_entries(weak_entry_t结构体中的inline数组)。

注:__weak修饰的变量会自动加入@auotreleasepool中延长生命周期至当前Runloop结束,否则创建即销毁。

weak实现的流程

声明变量默认属性是strong,weak需要用__weak标记。转汇编可知会调用objc_initWeak()初始化一个新的weak指针指向新的地址;然后内部调用storeWeak函数更新指针指向,真正实现了weak引用;释放时调用clearDeallocating遍历weak_table中,对象对应的weak_entry设为nil,然后删除entry,清理对象记录。

storeWeak函数实现

objc_storeWeak(&a, b)  // b的内存地址作为key,weak修饰的属性变量a的内存地址&a作为value注册到weak表中,当b为nil时,则把&a从weak表中删除。

总结:获取新旧对象的两个弱引用计数表,lock,调用weak_unregister_no_lock把weak指针从旧表中删除,调用weak_register_no_lock把weak指针添加到新对象的新表,并把新对象设置为弱引用状态,把地址赋给weak指针。

注:weak_entry_t提供了增加和删除weak指针的核心方法。

注:bool objc_object::sidetable_isWeaklyReferenced()函数实现中,result = it->second & SIDE_TABLE_WEAKLY_REFERENCED; // weak修饰的独享引用计数会被设置为SIDE_TABLE_WEAKLY_REFERENCED,虽然和strong共用一个sidetable,但不参与计数表的计算。setWeaklyReferenced同理。

weak和assign

修饰对象:对象 / 基本数据类型(也可以修饰对象)

赋值方式:赋值引用(地址) / 赋值 值

引用计数:都不影响引用计数

对象销毁后:自动置nil / 不变(野指针)

给现有类添加weak属性(AssociatedObject):引用计数器不变(OBJC_ASSOCIATION_ASSIGN),对象销毁后自动置nil(捕获对象的销毁时机dealloc方法)。

strong   

底层调用objc_storeStrong(*location, obj)函数,location是指针,obj是内存中的对象,内部实现:先判断id prev = *location,obj == *location retrun;,指针之前指向的地址不是当前对象→再执行retain引用计数+1→*location = obj,strong指针指向对象→objc_release(prev),释放prev指针对上一个对象的引用。

copy

copy关键字修饰的property属性的setter方法有个很重要的步骤:_copyProperty = [copyProperty copy];会做一次深拷贝,并把可变类型为不可变。

注:copy除了作为关键字,还有copy方法,不可变浅拷贝,可变深拷贝。而mutableCopy方法全是对对象进行深拷贝。

strong和copy的总结:修饰不可变类型时,都只做浅拷贝;修饰可变时,strong引用计数+1,copy深拷贝,指向不可变新对象。所以copy多用于修饰NSString、NSArray等有可变类型的不可变类型避免赋值成可变类型,造成影响

__unsafe_unretained

修饰非指针(修饰指针会造成野指针)的简单数据类型(如NSInteger,id,weak不行),等价assign,不需要weak的各种检查逻辑有一定的性能提升(对象释放时会查散列表删除weak指针)。可以理解在ARC时代,iOS系统向下兼容而存在的关键字。


AssociatedObject 关联对象

应用:1、可以在不改变源码的情况下为类添加实例变量(不属于类,存在一个全局表中);2、可以给分类添加成员变量(即实现category添加的属性的set/get方法):objc_setAssociatedObject和objc_getAssociatedObject;3、KVO中使用关联对象作为观察者可以避免循环引用。

底层原理:不是存在对象中的,而是通过AssociationsManager全局管理了一个AssociationsHashMap *_map,全局表中存储着所有对象的关联对象表ObjectAssociationMap,表内存着ObjectionAssociation关联对象的value和policy。通过运行时的方式添加和移除(=nil自动删除)。

关联对象的内存管理:policy枚举(assign,strong,copy,nonatomic,atomic),objc_setAssociatedObject时会指定policy,根据这个枚举对对象进行retain/copy或weak的方式进行内存管理。


相关问题

weak的实现原理,SideTable结构是怎样的

weak修饰的指针变量指向对象时不会增加对象的引用计数,而是把当前对象的所有weak指针维护在一个全局的hash表(SideTable)中,以对象为key,weak指针为value,对象释放时会遍历清除weak指针;storeWeak传入weak指针和对象时,会解除weak指针和旧对象的绑定,并完成与新的弱引用表的绑定。

全局有多个SideTable,每个SideTable内储存了许多对象的引用计数refcnts表和weak表,weak表中的weak_table_t结构体中的weak_entries,维护了对象所有的weak指针。


strong的实现

storeStrong函数:先判断指针之前指向的是否是当前对象,否:引用计数+1,*location = obj指针指向当前对象,objc_release(prev)释放指针之前指向的的对象。


@property

编译器会加入@synthesize自动合成为property_t = ivar+getter+setter,而且会自动生成成员变量ivar名字是property,而属性的名字就是_property。可以用@synthesize property = myProperty给变量自定义的属性名。

category中的@property:因为对象的内存布局在编译阶段已经确定,添加成员变量会破坏内存布局,所以分类中的@property只是添加到属性列表(properties),没有ivar和setter/getter。所以一般声明后,通过关联对象实现属性的setter/getter。

protocol中的@property:协议中其实也可以定义属性,需在.m中用@synthesize手动同步属性;或手动实现setter/getter方法。


@synthesize和@dynamic

@synthesize绑定ivar和存取器生成属性,一般编译器自动生成,可用于标记指定实例变量名;还可用于重写setter/getter,override父类属性和protocol中定义属性等,需要手动绑定存取器的情况。

@dynamic标记自己实现属性的setter/getter方法,而不使用自动合成


关联对象的原理和应用

AssociationsManager对象通过一张全局hash表维护,也是通过对象的指针地址作为key,value是所有的关联对象。多用于动态添加实例变量(分类中添加成员变量),其实变量不属于对象。

关联对象通过添加时的policy入参关键字决定内存管理方式(copy/strong/atomic等),设置成assign即是一个weak的关联对象了)


什么时候用__unsafe_unretained

__unsafe_unretained修饰的指针指向的对象在释放时,不需要weak的各种检查逻辑有一定的性能提升(对象释放时会查散列表删除weak指针),但要注意指针变量的作用范围,否则会造成野指针(适合局部临时使用如循环中)。可以理解为等价assign,iOS向下兼容的关键字。

你可能感兴趣的:(属性的声明和修饰)