iOS 属性关键字
Posted xiaoxiaobukuang
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了iOS 属性关键字相关的知识,希望对你有一定的参考价值。
一、分类
- 读写权限
- 原子性
- 引用计数
二、读写权限
readwrite
读写,readonly
只读。属性默认是readwrite
,支持读写。
1、readonly
属性只具有get
方法;
2、readwrite
属性同时具有set
和get
方法;
三、原子性
1、nonatomic
非原子的,atomiiac
原子的。属性默认是atomiac
,也就是原子性的。nonatomic
执行效率高。
2、atomic
读写安全,但效率低,不是绝对的安全,比如操作数组,增加或移除买这种情况下使用互斥锁来保证线程安全。
四、引用计数
1、strong/retain
- retain:一般是在
MRC
时使用,用来修饰对象,; - strong:一般是在
ARC
时使用,用来修饰对象,不能修饰基本数据类型;是每对这个属性引用一次,retainCount
就会+1,
2、assign/unsafe_unretained
- assign:即可以修饰基本数据类型,又可以修饰对象类型;当修饰对象类型时,不改变
retainCount
引用计数; - unsafe_unretained:只有在
MRC
时使用的比较频繁,在ARC时基本上不使用。unsafe_unretained
和weak
一样,表示的是对象的一种弱引用的关系,唯一的区别是:weak
修饰的对象被释放后,指向对象的指针会置空,也就是指向nil
,不会产生野指针;而unsafe_unretained
修饰的对象被释放后,指针不会置空,而是变成一个野指针,那么此时如果访问这个对象的话,程序就会Crash
,抛出BAD_ACCESS
的异常。
- 为什么我们不用assign去声明对象:
因为assign修饰的对象,在释放之后,指针的地址还是存在的,也就是说指针并没有被置为nil,造成野指针,访问野指针,会导致程序Crash。- 为什么可以用assign修饰基本数据类型:
因为基本数据类型是分配在栈上,栈的内存会由系统自己自动处理回收,不会造成野指针。
3、weak
weak
对属性引用时,retainCount
不变,只能修饰NSObject对象,不能修饰基本数据类型。主要用于避免循环引用。weak
的原理:runtime
维护了一个weak
表,用于存储指向某个对象的所有weak
指针。weak
表其实就是一个hash
(哈希)表,key
是所指对象的地址,value
是weak
指针的地址(这个地址是所指对象指针的地址)数组。所指对象在被释放之后会自动置为nil
。
4、copy
(1) 、浅拷贝
浅拷贝就是对内存地址的复制,让目标对象指针和源对象指向同一片内存空间,引用计数增加;
(2)、深拷贝
深拷贝让目标对象指针和源对象指针指向两片内存相同的内存空间。
可变对象的copy和mutableCopy都是深拷贝;
不可变对象的copy是浅拷贝,mutableCopy是深拷贝;
copy方法返回的都是不可变对象
五、@synthesize 和 @dynamic
@property
有两个对应的词,一个是 @synthesize
,一个是 @dynamic
。如果 @synthesize
和 @dynamic
都没写,那么默认的就是 @syntheszie var = _var
;
-
@synthesize
的语义是如果你没有手动实现setter
方法和getter
方法,那么编译器会自动为你加上这两个方法。 -
@dynamic
告诉编译器:属性的setter
与getter
方法由用户自己实现,不自动生成。(当然对于readonly
的属性只需提供getter
即可)。假如一个属性被声明为@dynamic var
,然后你没有提供@setter
方法和@getter
方法,编译的时候没问题,但是当程序运行到instance.var = someVar
,由于缺setter
方法会导致程序崩溃;或者当运行到someVar = var
时,由于缺getter
方法同样会导致崩溃。编译时没问题,运行时才执行相应的方法,这就是所谓的动态绑定。
以上是关于iOS 属性关键字的主要内容,如果未能解决你的问题,请参考以下文章