问题:项目中存在相同名称的分类,而且里面的方法名可以也可以相同,也就是不受与其相同名称的另一个分类的任何限制
一、什么情况下可以允许相同名称的分类存在
我们创建一个Demo工程,并创建一个ZYPerson
类,然后为这个类创建两个分类ZYPerson+Test
,当我们创建第二个的时候发现无法将分类创建到文件列表中
由于项目这两个相同的分类在不同的文件夹下,所以我创建一个object文件夹,然后在这个文件夹下创建另一个分类,发现是可以成功创建的
结论:在不同的文件夹下可以存在相同名称的分类,且两个相同名称的分类中也可以存在名称完全一样的方法
二、分类中同名的方法执行
在同名的两个ZYPerson+Test中写入同名的方法,一个实例方法,一个类方法,然后进行调用
运行工程,看下方法的执行情况
可以看出执行了与
main.m
在同一文件夹下的那个ZYPerson+Test
分类中的方法,那为什么会这样执行呢?这个时候我们可能会想到编译顺序,也许执行的是先编译的那个,那么我们看一下编译顺序结果却发现被执行的这个分类竟然是后编译的,下面我们调换一下这两个分类的编译顺序,把Object文件夹下的那个分类变为后编译的那个
再次运行工程,结果发现这个时候执行了Object文件夹下的那个分类的方法
结论:分类中同名方法的的执行顺序与编译顺序有关,但不是先编译的先执行,而是倒序。
三、分类中的方法是如何合并的,所有分类的类对象和元类对象是否都是同一个
OC中的对象有三种,每种内含有的内容如下图所示
分析:由图可见,一个类的实例方法存在于其类对象中,类方法存在于其元类对象中
当我们调用实例方法的时候,会通过实例对象的isa指针找到该实例对象的类对象,然后找到方法进行调用
当我们调用类方法时,会通过类对象的isa指针找到该类的元类对象,然后找到类方法进行调用
在我们调用分类方法的时候,调用实例方法并没有创建分类的实例,调用类方法也没有用分类去调用,而都是使用的原有的类
结论:由此可见所有分类的类对象和元类对象都是同一个
下面我们来探究一下这些分类的方法是如何合并的
我们分为编译和运行两种时期分别探究,首先看一下编译时期
选择其中一个分类进行反编译,这里选择的是ZYPerson+Eat,它里面有实例方法、类方法、协议和属性
在编译出的ZYPerson+Eat.cpp文件中搜索category_t,我们会发现如下图所示的结构体_category_t
从此处往下看,你会发现一些熟悉的元素
这里正是将分类中的方法,属性等信息生成相应的结构体变量,以方便赋值给_category_t结构体中的各个成员变量,再往下看你就会看到赋值的方法
由此可以看出,在编译时期:每个分类都是独立的结构体, 利用这种结构体存下分类方法,协议,属性等信息
运行时期
分析运行时的代码需要我们去苹果的资源网站去源码下载
一般选择最新的源码下载
下载后打开源码工程,然后搜索Category,可以看到确实存在这种结构体
然后按照下图的顺序去找关键部分的源码,objc-os.mm是runtime的入口,所以一般从这里开始去找相应模块,比如Category,然后再点进去看详细的方法
1.通过Runtime加载某个类的所有Category数据
2.把所有Category的方法、属性、协议数据,合并到一个大数组中
3.将合并后的分类数据,插入到原来的数据的前面