###@property相关 ####一、@property的本质是什么?ivar、getter、setter 是如何生成并添加到这个类中的? (1)@property=实例变量(ivar)+存取方法(getter和setter); (2)自动合成(autosynthesis)
####二、@protocol协议和category类别中是如何使用@property (1)只会生成setter和getter方法声明; (2)protocol中是希望遵守我协议的对象能实现该属性; (3)category需要使用关联对象:objc_setAssociatedObject和objc_getAssociatedObject
####三、@property中有哪些属性关键字? (1)原子性:nonatomic则不使用自旋锁,默认是atomic由编译器合成的方法会通过锁定机制确保其原子性。 (atomic不是绝对的线程安全,其实无论是否是原子性的只是针对于getter和setter而言,下面有代码例子) (2)读/写权限:readwrite(读写)、readonly(只读) (3)内存管理属性:assign、strong、weak、copy (4)方法名:getter= 、setter= 例如@property (nonatomic, getter=isOn) BOOL on; BOOL一般命名为isXXX; setter=一般用在特殊的情境下 new、init开头属性,要重新命名。 另外也可以用关键字进行特殊说明,来避免编译器报错:
@property(nonatomic, readwrite, copy, null_resettable) NSString *initBy;
- (NSString *)initBy __attribute__((objc_method_family(none)));
(5)不常用的:nonnull(不能为空)、nullable(可以为空)、null_resettable(setter可为空, gette不可为空)
一、nonnull 表示不能为空
@property (nonnull, nonatomic, copy) NSString *name;//写法一
@property (nonatomic, copy) NSString *__nonnull name;//写法二,小写时为两个下划线
@property (nonatomic, strong) NSString *_Nonnull name;//写法三,大写时为一个下划线
- (void)test{
self.name = nil;//系统会有警告不能给这个属性赋nil
// 这样子不提示
NSString *string = nil;
self.name = string;//这里系统不会识别到
}
二、nullable 表示可以为空
@property (nullable, nonatomic, copy) NSString *name;//写法一
@property (nonatomic, copy) NSString *__nullable name;//写法二,小写时为两个下划线
@property (nonatomic, strong) NSString *_Nullable name;//写法三,大写时为一个下划线
三、null_resettable setter可为空, gette不可为空
setter方法是nullable(可以赋空值),getter方法是nonnull(取值不能为空)
当看到由null_resettable修饰的属性时,就应该猜想这个属性的初始化采用了懒加载方式
######验证atomic不是绝对的线程安全
@interface ViewController ()
@property (atomic , strong) NSString *info;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
//A
dispatch_async(dispatch_get_global_queue(0, 0), ^{
while (1) {
self.info = @"a";
NSLog(@"A--info:%@", self.info);
}
});
//B
dispatch_async(dispatch_get_global_queue(0, 0), ^{
while (1) {
self.info = @"b";
NSLog(@"B--info:%@", self.info);
}
});
// 根据线程安全定义,如果atomic为线程安全A输出应该永远为A--info:a,B输出应该永远为B--info:b
// NSlog会有:A--info:b
}
@end
####四、@syncthesize和@dynamic分别有什么作用? (1)都是@property对应的词,默认是@syncthesize var=_var; (2)@syncthesize:编译器自动实现getter、setter方法; (3)@dynamic是告诉编译器,getter、setter方法由用户自己实现,不自动生成。
#####@synthesize 合成实例变量(隐藏创建一个_xxx)的规则,有以下几点:
(1)如果指定了成员变量的名称,会生成一个指定的名称的成员变量, (2)如果这个成员已经存在了就不再生成了. (3)如果是 @synthesize foo; 会生成一个名称为foo的成员变量. (4)默认是@syncthesize var=_var;
默认不写@syncthesize,是自动合成,如果不写@syncthesize,一下几种情况也不自动合成实例变量(隐藏创建一个_xxx)
(1)同时重写了 setter 和 getter 时 (2)重写了只读属性的 getter 时 (3)使用了 @dynamic 时 (4)在 @protocol 中定义的所有属性 (5)在 category 中定义的所有属性 (6)重载的属性
####五、ARC下,默认的属性修饰是什么? (1)基本数据类型的是:atomic,readwrite,assign (2)OC对象的是:atomic,readwrite,strong
####六.什么情况使用weak关键字?与assign有什么不同? (1)在ARC中,可能出现循环引用的时候,需要一段设置weak来解决,比如:delegate属性的修饰 (2)不同点:weak修饰属性,弱引用,不会保留新值,也不会释放旧值,如果旧值被摧毁这个属性赋值nil,继续使用不会闪退App assign 可以用非 OC 对象( CGFloat 或 NSlnteger 等),而 weak 必须用于 OC 对象
strong、assign、weak、copy等关键相关
####七、copy相关 #####1、怎么用copy关键字? (1)NSString、NSArray、NSDictionry等经常使用copy,因为他们有可变的,如果修饰strong把可变的赋值他们,他们会变成可变,为确保不会无意变动应该使用copy修饰; (2)Block在ARC下 赋值就是copy,栈复制到堆上; (3)copy出来的东西是不可变的,mutableCopy出来的东西是可变的
| 源对象类型 | 拷贝方法 | 副本对象类型 | 是否产生新对象 | 拷贝类型 |
|---|---|---|---|---|
| NS* | copy | NS* | 否 | 浅拷贝 |
| NS* | mutableCopy | NSMutable* | 是 | 深拷贝 |
| NSMutable* | copy | NS* | 是 | 深拷贝 |
| NSMutable* | mutableCopy | NSMutable* | 是 | 深拷贝 |
注:浅拷贝 == 指针拷贝;深拷贝 == 内容拷贝,深复制需要实现NSCoding协议,实现- (void)copyWithZone:(NSZone *)zone方法
#####2、父类实现深拷贝时,子类如何实现深度拷贝? (1)Person的copyWithZone里调用Person *p = [[[self class] alloc] init]; (2)Son的copyWithZone里调用Son *s = [super copyWithZone:zone]; (3)配置Son的属性
@implementation Person
- (id)copyWithZone:(NSZone *)zone
{
Person *p = [[[self class] alloc] init];
p. personId = self.personId;
return p;
}
@end
@implementation Son
- (id)copyWithZone:(NSZone *)zone
{
Son *s = [super copyWithZone:zone];
s.studentId = self.studentId;
return s;
}
@end
#####3、父类没有实现深拷贝时,子类如何实现深度拷贝? (1)Son的copyWithZone里调用Son *s = [[[self class] alloc] init]; (2)配置Person属性、配置Son的属性
@implementation Son
- (id)copyWithZone:(NSZone *)zone
{
Son *s = [[[self class] alloc] init];
s.personId = self. personId;
s.studentId = self.studentId;
return s;
}
@end
#####4、这个写法会出什么问题: @property (copy) NSMutableArray *array;? (1)copy出来的对象是不可变的,这个array如果操作add、rem、ins,会崩溃; (2)使用atomic(默认)原子性线程锁会影响性能。 nonatomic没有线程锁。