iOS相关知识(五)-- Category

1、分类的使用场景(也就是作用)

1、可以减少单个类的体积,降低耦合性,同一个类可以多人进行开发
2、可以为系统类添加分类进行拓展
3、模拟多继承
4、把静态库的私有方法公开

2、分类的底层结构

分类在编译的时候会生成一个结构体,在运行时就会把分类的方法列表、属性列表、列表等信息合并到类方法中。


分类的底层结构.png

3、category中有load方法吗? load方法什么时候调用的? load方法能继承吗?

1、category有load方法
2、load方法在runtime类、分类加载的时候调用且只调用一次
3、load是可以继承的 一般情况我们不会去主动调用load方法

4、Load与initialize的区别是什么? 他们在category中的调用顺序?出现继承是他们的调用过程?

调用时机不同
load方法在runtime类、分类加载的时候调用且只调用一次。
如果出现继承时 
首先会调用父类的load > 子类的load  >  分类的load
分类的load调用顺序是按照编译顺序来确定的。

initialize方法是在类第一次接收到消息时调用,
出现继承时父类的initialize方法可能会调用多次(父类的initialize调用多次并不代表父类会初始化多次,父类只会初始化一次)

调用方式不同
load方式是通过指针函数直接调用
initialize是通过objc_msgSend来调用

5、一个类的多个Category 都用同一个方法,当调用时会同时都调用吗?

答:不会
分类在编译的时候会生成一个结构体,在运行时就会把分类的方法列表、属性列表、
列表等信息合并到类方法中。

当调用一个方法的时候 根据消息转发机制的原理来看 只要找到方法即结束查找。
只调其中一个分类的方法,具体调用哪个分类的方法则根据编译的顺序决定,

6、Category 能否添加成员变量 如果可以 如何给category添加成员变量

不可以,分类是OC特有的语言,依赖于类。
分类的作用是在不改变原来的类内容的基础上,为类增加一些方法。
分类并不是真正的类,他没有isa 。
类在最初的时候就生成了一些基本的属性 比如 ivarlist  MethodList 
分类只会将自己的method 合并到主类 并不会影响主类的 IvarList
Ivar其实是一个objc_ivar的指针,objc_ivar是一个struct,其中包含变量名、变量类型。
Ivar实例变量所在内存区域初始化后不可更改,无法在运行时增加实例变量,所以分类是无法添加实例变量的。
虽然Category  不能直接添加成员变量 但是可以间接为Category 添加成员变量

使用 关联对象 为 Category 添加成员变量  例:为Person 的分类(FCPerson+Run)添加一个属性分类代码如下:

例子:使用runtime动态为分类添加成员变量

创建一个FCPerson 然后创建一个 Run 的分类

FCPerson+Run.h

#import "FCPerson.h"
NS_ASSUME_NONNULL_BEGIN
@interface FCPerson (Run)
@property (nonatomic,strong)NSString *run;
@end
NS_ASSUME_NONNULL_END

FCPerson+Run.m

#import "FCPerson+Run.h"
#import 

@implementation FCPerson (Run)

- (void)setRun:(NSString *)run{
    objc_setAssociatedObject(self, @selector(run), run, OBJC_ASSOCIATION_COPY_NONATOMIC); 
}

- (NSString *)run{
    return objc_getAssociatedObject(self, @selector(run));
}

@end

在VC里面调用如下

    FCPerson *person = [[FCPerson alloc]init];
    person.run = @"我要跑";
    NSLog(@"%@",person.run);

打印结果如下

2021-01-08 09:20:07.255413+0800 CategoryTest[3537:770282] 我要跑

注意:


1、分类可以添加属性,但是并不会自动生成成员变量及set/get方法。
因为category_t结构体中并不存在成员变量。
通过之前对对象的分析我们知道成员变量是存放在实例对象中的,并且编译的那一刻就已
经决定好了。而分类是在运行时才去加载的。那么我们就无法再程序运行时将分类的成员
变量中添加到实例对象的结构体中。因此分类中不可以添加成员变量。

2、如果分类中有和原有类同名的方法, 会优先调用分类中的方法, 就是说会忽略原有类的方法。

3、如果多个分类中都有和原有类中同名的方法, 那么调用该方法的时候执行谁由编译器决定,
编译器会执行最后一个参与编译的分类中的方法。

你可能感兴趣的:(iOS相关知识(五)-- Category)