Category方法合并方式

问题:项目中存在相同名称的分类,而且里面的方法名可以也可以相同,也就是不受与其相同名称的另一个分类的任何限制

UIImage+Bitmap
一、什么情况下可以允许相同名称的分类存在

我们创建一个Demo工程,并创建一个ZYPerson类,然后为这个类创建两个分类ZYPerson+Test,当我们创建第二个的时候发现无法将分类创建到文件列表中

由于项目这两个相同的分类在不同的文件夹下,所以我创建一个object文件夹,然后在这个文件夹下创建另一个分类,发现是可以成功创建的

ZYPerson+Test

结论:在不同的文件夹下可以存在相同名称的分类,且两个相同名称的分类中也可以存在名称完全一样的方法

二、分类中同名的方法执行

在同名的两个ZYPerson+Test中写入同名的方法,一个实例方法,一个类方法,然后进行调用

image.png

image.png

image.png

运行工程,看下方法的执行情况
image.png

可以看出执行了与main.m在同一文件夹下的那个ZYPerson+Test分类中的方法,那为什么会这样执行呢?这个时候我们可能会想到编译顺序,也许执行的是先编译的那个,那么我们看一下编译顺序
image.png

结果却发现被执行的这个分类竟然是后编译的,下面我们调换一下这两个分类的编译顺序,把Object文件夹下的那个分类变为后编译的那个
image.png

再次运行工程,结果发现这个时候执行了Object文件夹下的那个分类的方法
image.png

结论:分类中同名方法的的执行顺序与编译顺序有关,但不是先编译的先执行,而是倒序。

三、分类中的方法是如何合并的,所有分类的类对象和元类对象是否都是同一个

OC中的对象有三种,每种内含有的内容如下图所示


image.png

分析:由图可见,一个类的实例方法存在于其类对象中,类方法存在于其元类对象中
当我们调用实例方法的时候,会通过实例对象的isa指针找到该实例对象的类对象,然后找到方法进行调用
当我们调用类方法时,会通过类对象的isa指针找到该类的元类对象,然后找到类方法进行调用

在我们调用分类方法的时候,调用实例方法并没有创建分类的实例,调用类方法也没有用分类去调用,而都是使用的原有的类

结论:由此可见所有分类的类对象和元类对象都是同一个

下面我们来探究一下这些分类的方法是如何合并的

我们分为编译和运行两种时期分别探究,首先看一下编译时期

选择其中一个分类进行反编译,这里选择的是ZYPerson+Eat,它里面有实例方法、类方法、协议和属性


image.png

在编译出的ZYPerson+Eat.cpp文件中搜索category_t,我们会发现如下图所示的结构体_category_t


分类结构体

从此处往下看,你会发现一些熟悉的元素
生成_category_t结构体中的各个成员变量

这里正是将分类中的方法,属性等信息生成相应的结构体变量,以方便赋值给_category_t结构体中的各个成员变量,再往下看你就会看到赋值的方法


生成一个_category_t结构体

由此可以看出,在编译时期:每个分类都是独立的结构体, 利用这种结构体存下分类方法,协议,属性等信息
运行时期

分析运行时的代码需要我们去苹果的资源网站去源码下载
一般选择最新的源码下载

image.png

下载后打开源码工程,然后搜索Category,可以看到确实存在这种结构体


image.png

然后按照下图的顺序去找关键部分的源码,objc-os.mm是runtime的入口,所以一般从这里开始去找相应模块,比如Category,然后再点进去看详细的方法


image.png

1.通过Runtime加载某个类的所有Category数据
2.把所有Category的方法、属性、协议数据,合并到一个大数组中

3.将合并后的分类数据,插入到原来的数据的前面

你可能感兴趣的:(Category方法合并方式)