- 分类
- 扩展
- 代理(Delegate)
- 通知(NSNotification)
- KVO (Key-value observing)
- KVC (Key-value coding)
- 属性关键字
1、分类web
学习的圈子特别重要!!
附上一份收集的各大厂面试题(附答案) ! 群文件直接获取
各大厂面试题面试
做为一个开发者,有一个学习的氛围跟一个交流圈子特别重要,这是一个个人iOS交流群:761407670 进群密码‘博客’,无论你是小白仍是大牛欢迎入驻 ,分享BAT,阿里面试题、面试经验,讨论技术, 你们一块儿交流学习成长!设计模式
2、扩展数组
-
1.通常用扩展作什么?
声明私有属性,声明方法(没什么意义),声明私有成员变量
-
2.扩展的特色
编译时决议,只能以声明的形式存在,多数状况下寄生在宿主类的.m中,不能为系统类添加扩展。
3、代理(Delegate)安全
代理是一种设计模式,以@protocol形式体现,通常是一对一传递。
通常以weak关键词以规避循环引用。bash
4、通知(NSNotification)
使用观察者模式来实现的用于跨层传递信息的机制。传递方式是一对多的。函数
5、KVO (Key-value observing)
KVO是观察者模式的另外一实现。
使用了isa混写(isa-swizzling)来实现KVO学习
使用setter方法改变值KVO会生效,使用setValue:forKey即KVC改变值KVO也会生效,由于KVC会去调用setter方法ui
- (void)setValue:(id)value { [self willChangeValueForKey:@"key"]; [super setValue:value]; [self didChangeValueForKey:@"key"]; } 复制代码
-
那么经过直接赋值成员变量会触发KVO吗?
不会,由于不会调用setter方法,须要加上
willChangeValueForKey和didChangeValueForKey方法来手动触发才行
6、KVC(Key-value coding)atom
-(id)valueForKey:(NSString *)key;
-(void)setValue:(id)value forKey:(NSString *)key; 复制代码
KVC就是指iOS的开发中,能够容许开发者经过Key名直接访问对象的属性,或者给对象的属性赋值。而不须要调用明确的存取方法。这样就能够在运行时动态地访问和修改对象的属性。而不是在编译时肯定,这也是iOS开发中的黑魔法之一。不少高级的iOS开发技巧都是基于KVC实现的
当调用setValue:属性值 forKey:@”name“的代码时,,底层的执行机制以下:
- 程序优先调用set<Key>:属性值方法,代码经过setter方法完成设置。注意,这里的<key>是指成员变量名,首字母大小写要符合KVC的命名规则,下同
- 若是没有找到setName:方法,KVC机制会检查+ (BOOL)accessInstanceVariablesDirectly方法有没有返回YES,默认该方法会返回YES,若是你重写了该方法让其返回NO的话,那么在这一步KVC会执行setValue:forUndefinedKey:方法,不过通常开发者不会这么作。因此KVC机制会搜索该类里面有没有名为<key>的成员变量,不管该变量是在类接口处定义,仍是在类实现处定义,也不管用了什么样的访问修饰符,只在存在以<key>命名的变量,KVC均可以对该成员变量赋值。
- 若是该类即没有set<key>:方法,也没有_<key>成员变量,KVC机制会搜索_is<Key>的成员变量。
- 和上面同样,若是该类即没有set<Key>:方法,也没有_<key>和_is<Key>成员变量,KVC机制再会继续搜索<key>和is<Key>的成员变量。再给它们赋值。
- 若是上面列出的方法或者成员变量都不存在,系统将会执行该对象的setValue:forUndefinedKey:方法,默认是抛出异常。
即若是没有找到Set<Key>方法的话,会按照_key,_iskey,key,iskey的顺序搜索成员并进行赋值操做。
若是开发者想让这个类禁用KVC,那么重写+ (BOOL)accessInstanceVariablesDirectly方法让其返回NO便可,这样的话若是KVC没有找到set<Key>:属性名时,会直接用setValue:forUndefinedKey:方法。
当调用valueForKey:@”name“的代码时,KVC对key的搜索方式不一样于setValue:属性值 forKey:@”name“,其搜索方式以下:
- 首先按get<Key>,<key>,is<Key>的顺序方法查找getter方法,找到的话会直接调用。若是是BOOL或者Int等值类型, 会将其包装成一个NSNumber对象。
- 若是上面的getter没有找到,KVC则会查找countOf<Key>,objectIn<Key>AtIndex或<Key>AtIndexes格式的方法。若是countOf<Key>方法和另外两个方法中的一个被找到,那么就会返回一个能够响应NSArray全部方法的代理集合(它是NSKeyValueArray,是NSArray的子类),调用这个代理集合的方法,或者说给这个代理集合发送属于NSArray的方法,就会以countOf<Key>,objectIn<Key>AtIndex或<Key>AtIndexes这几个方法组合的形式调用。还有一个可选的get<Key>:range:方法。因此你想从新定义KVC的一些功能,你能够添加这些方法,须要注意的是你的方法名要符合KVC的标准命名方法,包括方法签名。
- 若是上面的方法没有找到,那么会同时查找countOf<Key>,enumeratorOf<Key>,memberOf<Key>格式的方法。若是这三个方法都找到,那么就返回一个能够响应NSSet所的方法的代理集合,和上面同样,给这个代理集合发NSSet的消息,就会以countOf<Key>,enumeratorOf<Key>,memberOf<Key>组合的形式调用。
- 若是尚未找到,再检查类方法+ (BOOL)accessInstanceVariablesDirectly,若是返回YES(默认行为),那么和先前的设值同样,会按_<key>,_is<Key>,<key>,is<Key>的顺序搜索成员变量名,这里不推荐这么作,由于这样直接访问实例变量破坏了封装性,使代码更脆弱。若是重写了类方法+ (BOOL)accessInstanceVariablesDirectly返回NO的话,那么会直接调用valueForUndefinedKey:方法,默认是抛出异常。
7、属性关键字
1.读写权限:readonly,readwrite(默认)
2.原子性: atomic(默认),nonatomic。atomic读写线程安全,但效率低,并且不是绝对的安全,好比若是修饰的是数组,那么对数组的读写是安全的,但若是是操做数组进行添加移除其中对象的还,就不保证安全了。
3.引用计数:
- retain/strong
- assign:修饰基本数据类型,修饰对象类型时,不改变其引用计数,会产生悬垂指针,修饰的对象在被释放后,assign指针仍然指向原对象内存地址,若是使用assign指针继续访问原对象的话,就可能会致使内存泄漏或程序异常
- weak:不改变被修饰对象的引用计数,所指对象在被释放后,weak指针会自动置为nil
- copy:分为深拷贝和浅拷贝
浅拷贝:对内存地址的复制,让目标对象指针和原对象指向同一片内存空间会增长引用计数
深拷贝:对对象内容的复制,开辟新的内存空间
可变对象的copy和mutableCopy都是深拷贝
不可变对象的copy是浅拷贝,mutableCopy是深拷贝
copy方法返回的都是不可变对象