分类(Category)和类扩展(Extension)
什么是分类,什么是类扩展,他们分别用在什么场合,各有哪些不能被替代的特点?
分类(Category):它是在运行时决议的,我们可以给类或者系统类添加实例方法方法。我们添加的实例方法,会被动态的添加到类结构里面的methodList
列表里面。
类扩展(Extension):看起来很像一个匿名的category,但是extension和有名字的category几乎完全是两个东西。
extension在编译时决议,它就是类的一部分,在编译期和头文件里的@interface
以及实现文件里的@implement
一起形成一个完整的类,它伴随类的产生而产生,亦随之一起消亡。extension一般用来隐藏类的私有信息,你必须有一个类的源码才能为一个类添加extension,所以你无法为系统的类比如NSString添加extension。
分类用到的场合
1.可以为类添加方法
2.在runtime
可以为类添加属性
在1这样一种情况大家用的应该比较多,可以在不修改原来类的基础上,为一个类扩展方法。最主要的用法,给系统自带的类扩展方法。
比方说,在工程里每个UIViewController里都想要做同样一件事情,如果一个一个的加,浪费时间不说,以后增加了新页面,还需要添加方法。
这就可以用到分类,想要在每个页面都执行的代码,可以写在这些页面的父类中。我们可以把代码写在UIViewController中。
但是它也有局限性:
1.分类中只能添加方法,不能增加成员变量。(可以添加属性)
2.分类中可以访问原来类中的成员变量,但是只能访问@protect和@public形式的变量。如果想要访问本类中的私有变量,分类和子类一样,只能通过方法来访问。
为什么不能增加成员变量?
在objc_class
结构体中,ivars
是objc_ivar_list
(成员变量列表)指针;methodLists
是指向objc_method_list
指针的指针。在Runtime中,objc_class
结构体大小是固定的,不可能往这个结构体中添加数据,只能修改。所以ivars
指向的是一个固定区域,只能修改成员变量值,不能增加成员变量个数。methodList
是一个二维数组,所以可以修改*methodLists
的值来增加成员方法,虽没办法扩展methodLists指向的内存区域,却可以改变这个内存区域的值(存储的是指针)。
简单的我想讲一下如何实现为分类添加属性,但是不会生成_变量
(带下划线的成员变量),也不会生成添加属性的getter和setter方法的实现:
比如,为UIViewController添加显示的时间这样一个属性。
static char *TimeKey = "TimeKey";
-(void)setTime:(NSTimeInterval)time {
objc_setAssociatedObject(self, TimeKey, cloudox, OBJC_ASSOCIATION_COPY_NONATOMIC);
}
-(NSString *)time{
return objc_getAssociatedObject(self, TimeKey);
}
注意,要导入头文件。
#import <objc/runtime.h>
分类的执行优先级
1.在本类和分类有相同的方法时,优先调用分类的方法再调用本类的方法。
2.如果有两个分类,他们都实现了相同的方法,如何判断谁先执行?分类执行顺序可以通过targets,Build Phases,Complie Source进行调节,注意执行顺序是从上到下的。(只有两个相同方法名的分类)
extension
类扩展(extension)是category的一个特例,有时候也被称为匿名分类。他的作用是为一个类添加一些私有的成员变量和方法。
类扩展能写点啥?和分类不同,类扩展即可以声明成员变量又可以声明方法。
类扩展听上去很复杂,但其实我们很早就认识他了。
#import <UIKit>
#import "LoginViewController.h"
@interface LoginViewController ()
@property (nonatomic, strong) IBOutlet UITextField* nameTF;
@property (nonatomic, strong) IBOutlet UITextField* pwdTF;
- (IBAction)login:(id)sender;
@end
@implementation LoginViewController
- (void)viewDidLoad {
[super viewDidLoad];
}
- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
}
- (IBAction)login:(id)sender {
}
@end
是不是很熟悉了,2333。
对比看一下继承自NSObject的类我们会发现没有上面的代码块。
@interface LoginViewController()//这就是类扩展的写法
@end
类扩展可以定义在.m文件中,这种扩展方式中定义的变量都是私有的,也可以定义在.h文件中,这样定义的代码就是共有的,类扩展在.m文件中声明私有方法是非常好的方式。
类扩展中添加的新方法,一定要实现。category中没有这种限制。
比较category和extension, 总结来自于(VV木公子 简书作者):
extension在编译期决议,它就是类的一部分,但是category则完全不一样,它是在运行期决议的。extension在编译期和头文件里的@interface
以及实现文件里的@implement
一起形成一个完整的类,它、extension伴随类的产生而产生,亦随之一起消亡。
extension一般用来隐藏类的私有信息,你必须有一个类的源码才能为一个类添加extension,所以你无法为系统的类比如NSString
添加extension,除非创建子类再添加extension。而category不需要有类的源码,我们可以给系统提供的类添加category。
extension可以添加实例变量,而category不可以。
extension和category都可以添加属性,但是category的属性不能生成成员变量和getter、setter方法的实现。
额外的补充,Objective-C中的点语法说明:
如果点表达式出现在 "=" 左边,该属性名称的setter
方法将被调用。如果点表达式出现在右边,该属性名称的getter方法将被调用。所以在oc中点表达式其实就是调用对象的setter
和getter
方法的一种快捷方式。
@synthesize
还有一个作用,可以指定与属性对应的实例变量,例如@synthesize var = xxxx;
那么self.var
其实是操作的实例变量而是xxxx
,不是_var
了。