iOS面试题---Objective_C语言特性:分类、扩展、代理、通知、KVO、KVC、属性
战地4directx error,51la站长统计,长城小报
-
分类
-
扩展
-
代理(delegate)
-
通知(nsnotification)
-
kvo (key-value observing)
-
kvc (key-value coding)
-
属性关键字
转自:https://www.cnblogs.com/IOSkf/p/12869132.html
一、分类
学习的圈子特别重要!!
附上一份收集的各大厂面试题(附答案) ! 群文件直接获取
各大厂面试题
作为一个开发者,有一个学习的氛围跟一个交流圈子特别重要,这是一个我的ios交流群:761407670 进群密码‘博客’,不管你是小白还是大牛欢迎入驻 ,分享bat,阿里面试题、面试经验,讨论技术, 大家一起交流学习成长!
二、扩展
-
1.一般用扩展做什么?
声明私有属性,声明方法(没什么意义),声明私有成员变量
-
2.扩展的特点
编译时决议,只能以声明的形式存在,多数情况下寄生在宿主类的.m中,不能为系统类添加扩展。
三、代理(delegate)
代理是一种设计模式,以@protocol形式体现,一般是一对一传递。
一般以weak关键词以规避循环引用。
四、通知(nsnotification)
使用观察者模式来实现的用于跨层传递信息的机制。传递方式是一对多的。
五、kvo (key-value observing)
kvo是观察者模式的另一实现。
使用了isa混写(isa-swizzling)来实现kvo
使用setter方法改变值kvo会生效,使用setvalue:forkey即kvc改变值kvo也会生效,因为kvc会去调用setter方法
- (void)setvalue:(id)value
{
[self willchangevalueforkey:@"key"];
[super setvalue:value];
[self didchangevalueforkey:@"key"];
}
复制代码
-
那么通过直接赋值成员变量会触发kvo吗?
不会,因为不会调用setter方法,需要加上
willchangevalueforkey和didchangevalueforkey方法来手动触发才行
六、kvc(key-value coding)
-(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:方法,默认是抛出异常。
七、属性关键字
1.读写权限:readonly,readwrite(默认)
2.原子性: atomic(默认),nonatomic。atomic读写线程安全,但效率低,而且不是绝对的安全,比如如果修饰的是数组,那么对数组的读写是安全的,但如果是操作数组进行添加移除其中对象的还,就不保证安全了。
3.引用计数:
-
retain/strong
-
assign:修饰基本数据类型,修饰对象类型时,不改变其引用计数,会产生悬垂指针,修饰的对象在被释放后,assign指针仍然指向原对象内存地址,如果使用assign指针继续访问原对象的话,就可能会导致内存泄漏或程序异常
-
weak:不改变被修饰对象的引用计数,所指对象在被释放后,weak指针会自动置为nil
-
copy:分为深拷贝和浅拷贝
浅拷贝:对内存地址的复制,让目标对象指针和原对象指向同一片内存空间会增加引用计数
深拷贝:对对象内容的复制,开辟新的内存空间
可变对象的copy和mutablecopy都是深拷贝
不可变对象的copy是浅拷贝,mutablecopy是深拷贝
copy方法返回的都是不可变对象
如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!!
点击进行留言回复
相关文章:
-
-
-
-
-
iOS实现图片自动切换效果
本文实例为大家分享了ios实现图片自动切换的具体代码,供大家参考,具体内容如下#import "viewcontroller.h"#define imagevi...
[阅读全文]
-
-
-
-
-
-
网友评论