一、 分类和类扩展区别
1. 分类实现原理
- Category编译之后的底层结构是struct category_t,里面存储着分类的对象方法、类方法、属性、协议信息
- 在程序运行的时候,runtime会将Category的数据,合并到类信息中(类对象、元类对象中)
2. Category和Class Extension的区别是什么?
- Class Extension在编译的时候,它的数据就已经包含在类信息中
- Category是在运行时,才会将数据合并到类信息中
二、 分类为啥不能添加成员变量
先看Category的底层结构
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; // 属性列表
};
1.从结构体可以知道,有
属性列表
,所以分类可以声明属性
,但是分类只会生成该属性对应的get
和set
的声明
,没有去实现该方法
。
2.结构体没有成员变量列表
,所以不能声明成员变量。
1. Category的加载处理过程
- 1.通过Runtime加载某个类的所有Category数据
- 2.把所有Category的方法、属性、协议数据,合并到一个大数组中,后面参与编译的Category数据,会在数组的前面
- 3.将合并后的分类数据(方法、属性、协议),插入到类原来数据的前面
三、关联对象给分类添加属性
代码实现如下
- Student+Extern.m
#import "Student+Extern.h"
#import <objc/runtime.h>
static NSString *nameKey = @"nameKey"; //定义一个key值
@implementation Student (Extern)
- (void)setName:(NSString *)name {
objc_setAssociatedObject(self, &nameKey, name, OBJC_ASSOCIATION_COPY);
}
- (NSString *)name {
return objc_getAssociatedObject(self, &nameKey);
}
@end
外界调用
Student *stu = [[Student alloc] init];
stu.name = @"乔碧萝";
NSLog(@"name = %@",stu.name);
运行结果 - 关联成功
但是注意,以上代码仅仅是手动实现了setter/getter方法,但调用_成员变量依然报错。
更多:iOS面试题合集