分类category
category
的介绍
Category
是oc2.0添加的语言特性,它的主要作用是在不改变本类下,动态地给这个类添加方法,除此之外Category
还可以将一个类进行拆分在不同文件下进行管理、甚至可以模拟实现多继承。若Category
添加的方法在基类中已经存在,则会‘覆盖’基类的同名方法,这里的覆盖并不是真正的覆盖,而是在方法列表中取用了第一个同名方法名。
分类、类、父类中实例方法和类方法调用顺序
分类、类和父类的方法调用顺序,在我另外一片关于iOS底层实现的文章中,分析了类的本质和方法存放位置,不清楚的同学可以先看下这篇文章,通过之前的分析我们知道方法都是将方法转换成objc_msgSeng(obj,SEL)
的形式进行消息发送的,首先会通过查找obj
的isa找到其类对象,然后在类对象的方法列表中查找SEL
方法,如果能找到的话就调用类对象的方法,如果类方法中不存在该方法就通过supclass指针找到其父类对象,在它父类对象的方法列表中查找该方法,这样一层层的查找实例方法,类似的类方法就是去其元类对象找那个查找而已,流程是一样的。
知道了类和父类的方法调用顺序后,我们就好奇category
中的方法是怎么调用的,它的结构体是怎样的。这里我先给大家打印下category
在c语言中的结构体以及项目中写的真是分类对应取值
struct _category_t {
const char *name;
struct _class_t *cls;
const struct _method_list_t *instance_methods;
const struct _method_list_t *class_methods;
const struct _protocol_list_t *protocols;
const struct _prop_list_t *properties;
};
static struct _category_t _OBJC_$_CATEGORY_PQStudent_$_test __attribute__ ((used, section ("__DATA,__objc_const"))) =
{
"PQStudent",
0, // &OBJC_CLASS_$_PQStudent,
(const struct _method_list_t *)&_OBJC_$_CATEGORY_INSTANCE_METHODS_PQStudent_$_test,
(const struct _method_list_t *)&_OBJC_$_CATEGORY_CLASS_METHODS_PQStudent_$_test,
0,
0,
};
分析结构体,不难发现,结构体中有六个字段组成,分别是本类名称、本垒的类机构体、实例方法列表、类方法列表、协议列表、属性列表,所以分类是不允许添加成员变量的,要想添加属性,必须通过添加关联实现setter和getter方法,然后在加载分类的时候,系统会遍历分类列表将各个分类的实例方法和类方法插入到类方法列表中和元类方法列表中,并且会根据先后编译顺序,以此排列,所以如果存在同名函数并不是覆盖,而是取用的时候有先后顺序。总结分类、类、父类的调用顺序是,先调用分类中的方法,然后调用类中的方法最后调用父类中的方法,分类如果有同名的方法,后编译的分类中方法先调用。
在iOS开发过程中,分类category可以用来扩展一个类的功能,让这个本类功能更强大,而使用分类的方式进行管理,在一定前提下会比使用子类方式更好管理,这个前提是在分类以及其下面的方法命名要规范,另外分类不能直接添加成员属性,如果一定要添加的话可以使用objc_getAssociatedObject
和objc_setAssociatedObject
方法绑定get和set方法,这里就不讲这两个方法的使用了。
分类、子类、父类的加载顺序
oc在编译的时候,就会加载所有的类,加载的顺序依次是父类->子类->分类,在加载完类后会调用load方法,我们可以在load方法中打印验证加载顺序,这里简单讲诉下,load
方法的调用原理,load
方法不同于其他类方法或者实例方法是通过消息机制objc_msgSend
进行调用的,实际上load
方法是直接调用load
所造内存指针指向的方法,看源码可以发现,load
方法的调用是递归进行的,在调用自身之前会先调用父类的load方法。
2019-02-27 16:36:44.842874+0800 swizzleDemo[13546:18542400] wenpq...本类加载
2019-02-27 16:36:44.843789+0800 swizzleDemo[13546:18542400] wenpq...子类加载
2019-02-27 16:36:44.844219+0800 swizzleDemo[13546:18542400] wenpq...分类加载
load方法只会加载一次,并且不同于initialize
这种懒加载方法,所以load是进行swizzle的很好方法。
swizzle使用方法
为了保证swizzle只执行一次,所以放在load方法中执行,其中originMethod
代表旧方法,newMethod
代表新方法,在方法交换前先用class_addMethod
添加方法,防止newMethod
不存在导致崩溃,class_addMethod
方法在newMethod
存在的情况下,仍会返回NO.
+ (void)load {
NSLog(@"wenpq...分类加载");
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
Method originMethod = class_getInstanceMethod([self class], @selector(work));
Method newMethod = class_getInstanceMethod([self class], @selector(swizzle_work));
BOOL addSuccess = class_addMethod([self class], @selector(swizzle_work), method_getImplementation(newMethod), method_getTypeEncoding(newMethod));
// 如果方法存在,也会添加失败
if (addSuccess) {
//用刚add的新方法replace旧方法
class_replaceMethod([self class], @selector(work), method_getImplementation(originMethod), method_getTypeEncoding(originMethod));
} else {
method_exchangeImplementations(originMethod, newMethod);
}
});
}
- (void)swizzle_work {
NSLog(@"wenpq...分类work");
[self swizzle_work];//调用原来的方法
}
目前只列出swizzle的是使用方法,之后会补充常用使用场景
项目demo地址