1、Category的实现原理

  • Category编译之后的底层结构是struct category_t,里面存储着分类的对象方法、类方法、属性、协议信息
  • 在程序运行的时候,runtime会将Category的数据,合并到类信息中(类对象、元类对象中)

2、Category和Class Extension的区别是什么?

  • Class Extension在编译的时候,它的数据就已经包含在类信息中
  • Category是在运行时,才会将数据合并到类信息中

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

  • 有load方法
  • load方法在runtime加载类、分类的时候调用
  • load方法可以继承,但是一般情况下不会主动去调用load方法,都是让系统自动调用

4、load、initialize方法的区别什么?它们在category中的调用的顺序?以及出现继承时他们之间的调用过程?

  • 调用方式:load是根据函数地址直接调用;initialize是通过objc_msgSend调用
  • 调用时刻:load是runtime加载类、分类的时候调用(只会调用一次);initialize是类第一次接收到消息的时候调用,每一个类只会initialize一次(父类的initialize可能会被调用多次)
  • 调用顺序:load是先调用类的load(先编译的类,优先调用load,调用子类的load前,会优先调用父类的load),再调用分类的load按编译顺序;initialize是先初始化父类,再初始化子类(有可能调用父类的initialize方法)

5、Category能否添加成员变量?如果可以,如何给Category添加成员变量?

不能直接给Category添加成员变量,但是可以间接实现Category有成员变量的效果

接下来是源码分析:

我们创建一个分类,然后转变成c++代码,xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc Person+Test.m

这是转变成c++代码看到的底层结构,我们再直接看objc4源码

  • objc-os.mm
  • _objc_init_image
  • _read_images

  • objc-runtime-new.mm 接下来在这个文件

  • _getObjc2CategoryList

  • remethodizeClass

  • attachCategories

  • attachLists

  • memmove() && memcpy() // 将分类的方法添加到类方法前

  1. 通过Runtime加载某个类的所有Category数据

  2. 把所有Category的方法、属性、协议数据,合并到一个大数组中 (后面参与编译的Category数据,会在数组的前面)

  3. 将合并后的分类数据(方法、属性、协议),插入到类原来数据的前面

接下来是load方法有关

+load源码解读顺序:objc-os.mm

  • _objc_unload_image

  • prepare_load_methods -> schedule_class_load - >schedule_class_load - > add_class_to_loadable_list

  • call_load_methods -> call_class_loads - > call_category_loads->(*load_method)(cls, SEL_load);

  1. 每个类、分类的+load,在程序运行过程中只调用一次

  2. 先调用类的+load ,按照编译先后顺序调用(先编译,先调用), 调用子类的+load之前会先调用父类的+load

  3. 再调用分类的+load

+load方法是根据方法地址直接调用,并不是经过objc_msgSend函数调用.

+initialize方法源码解读顺序:objc-os.mm

  • objc-msg-arm64.s(直接搜索)因为是汇编语言,不能直接看源码,所以我们根据他是调用方法,所以可以跳过这一步
  • class_getInstanceMethod
  • lookUpImpOrNil

  • lookUpImpOrForward

  • initializeAndLeaveLocked-> initializeAndMaybeRelock-> initializeNonMetaClass->_setThisThreadIsInitializingClass

  1. 先调用父类的+initialize,再调用子类的+initialize

category添加成员变量

@property (copy, nonatomic) NSString *name;在分类里面是添加了申明,但是没有实现。我们可以自己实现,你可以自定义全局一个字典,将对象的地址作为key,存取,但是这样也不是很好,我们用关联对象处理。

- (void)setName:(NSString *)name
{objc_setAssociatedObject(self, @selector(name), name, OBJC_ASSOCIATION_COPY_NONATOMIC);
}- (NSString *)name
{// 隐式参数// _cmd == @selector(name)return objc_getAssociatedObject(self, _cmd);
}

objc_AssociationPolicy

对应的修饰符

OBJC_ASSOCIATION_ASSIGN

assign

OBJC_ASSOCIATION_RETAIN_NONATOMIC

strong, nonatomic

OBJC_ASSOCIATION_COPY_NONATOMIC

copy, nonatomic

OBJC_ASSOCIATION_RETAIN

strong, atomic

OBJC_ASSOCIATION_COPY

copy, atomic

我们直接看源码分析,源码解读在objc-references.mm,实现关联对象技术的核心对象有

  • AssociationsManager
  • AssociationsHashMap
  • ObjectAssociationMap
  • ObjcAssociation

这是对应的结构,接下来我们直接用图讲解吧。

AssociationsManager

AssociationsHashMap *_map;

AssociationsHashMap

disguised_ptr_t

ObjectAssociationMap

disguised_ptr_t

ObjectAssociationMap

...

...

AssociationsMap

void *

ObjectAssociation

void *

ObjectAssociation

...

...

AssociationsMap

void *

ObjectAssociation

void *

ObjectAssociation

...

...

  1. 关联对象并不是存储在被关联对象本身内存中
  2. 关联对象存储在全局的统一的一个AssociationsManager中
  3. 设置关联对象为nil,就相当于是移除关联对象

面试题-Category(分类)相关推荐

  1. 【Groovy】MOP 元对象协议与元编程 ( 方法注入 | 使用 Category 分类进行方法注入的优缺点 )

    文章目录 一.使用 Category 分类进行方法注入的优点 二.使用 Category 分类进行方法注入的缺点 一.使用 Category 分类进行方法注入的优点 之前的博客 [Groovy]MOP ...

  2. 【Groovy】MOP 元对象协议与元编程 ( 方法注入 | 使用 Category 分类注入方法 )

    文章目录 一.方法注入 二.使用 Category 分类注入方法 三.完整代码示例 一.方法注入 在之前的博客中 , 主要是使用 Groovy 元编程 拦截方法 , 改变方法的实现 ; 使用元编程还可 ...

  3. Category 分类、类别 总结 - iOS

    分类作用 分类:可以给某一个类扩充一些方法(不修改原来类的代码) 格式 // 声明 @interface Person (分类名称)@end// 实现 @implementation Person ( ...

  4. 20210514面试题自我分类及总结(一)

    20210514面试题自我分类及总结 文章目录 20210514面试题自我分类及总结 前言 一.html 1.页面导入样式时,使用link和@import有什么区别? 二.css 1.双飞燕布 2.标 ...

  5. Category(分类/类目)、Protocol(协议)、Extension(扩展/延展)

    Category(分类/类目): 利用Objective-C的动态运行时分配机制,Category提供了一种比继承更为简洁的方法来对class进行扩展,无需创建对象类的子类就能为现有的类添加新方法,可 ...

  6. woocommerce 分类到菜单_Woocommerce教程:添加和编辑产品Category分类

    在使用WordPress的woocommerce商城插件制定商城的时候, 如果你公司拥有众多产品, 那么应该需要给这些产品添加Category分类, 以便分类管理产品, 让网站访客更快地找到自己想要的 ...

  7. Android大厂面试题系统分类从基础到困难(BATJ,蚂蚁金服,字节跳动,网易云,QQ音乐...)

    前言 早在2017年我们就建了第一个进击BAT的Android开发进阶交流群,两年期间很多群友都分享了自己的Android面试经历.其中就有很多群友已经斩获蚂蚁金服,天猫,高德,盒马等阿里系offer ...

  8. 软件测试工程师面试题答案分类详解-深圳某老牌培训机构内部绝密文件!绝密文件!绝密文件!

    目录:导读 一:前言:浅谈软件测试工程面试 二.数据库和LINUX最常见面试题(重点) 三.功能测试以及项目相关的常见面试题(重点) 四.接口测试常见面试题 五.Python以及自动化测试常见面试题( ...

  9. 软件测试工程师面试题答案分类详解-一家老牌培训机构内部绝密文件!

    目录:导读 一:前言:浅谈软件测试工程面试 二.数据库和LINUX最常见面试题(重点) 三.功能测试以及项目相关的常见面试题(重点) 四.接口测试常见面试题 五.Python以及自动化测试常见面试题( ...

最新文章

  1. 扒皮下京东首页楼层图标的动画效果实现方式
  2. 资源监控工具Spotlight-使用说明
  3. 计算机操作系统:存储器的管理
  4. 微信小程序wx.request内调用setData方法
  5. Java经典面试题整理及答案详解(二)
  6. 如何通过像素点找到世界坐标_如何通过阅读来找到自己理论研究的“视域”?...
  7. debian安装java jdk_Linux(Centos、Debian)之安装Java JDK及注意事项(转)
  8. 比尔盖茨:若新冠病毒疫苗有效 富裕国家在2021年底恢复正常
  9. ElasticSearch 2 (16) - 深入搜索系列之近似度匹配
  10. jsarraybufferdaya释放_聊聊JS的二进制家族:Blob、ArrayBuffer和Buffer
  11. Pandas:DataFrame对象的基础操作
  12. 常用公共数据集----数据获取
  13. blender风格化草地
  14. JavaEE中级.20190615.易买网项目(一).隐藏域.
  15. 别让你的项目,毁在单片机选型上
  16. 用地图说话:在商业分析与演示中运用Excel数据地图(全彩)
  17. 使用PEG估值法简单选股(1)
  18. 186.MultiAutoCompleteTextView
  19. K线形态识别—K线反转形态之底部反转形态
  20. 服务器销售工资如何计算公式,拿提成的工资怎么算的有公式吗

热门文章

  1. 单变量线性回归中的梯度下降法求解代价函数的最小值
  2. 读取xml忽略dtd验证
  3. Chosen中选择项的更新
  4. JS中setTimeout()的使用方法具体解释
  5. __GLOBAL__I_a in libUPPayPlugin.a(UPPasswordTool.o)
  6. 常见错误: 创建 WCF RIA Services 类库后, 访问数据库出错
  7. Nginx与Redis解决高并发问题
  8. vue 过滤器的使用(解决forEach遇到的问题)
  9. 【sql server】“因为数据库正在使用,所以无法获得对数据库的独占访问权” 解决方案汇总
  10. “〜”(波浪号/波浪形/旋转)CSS选择器是什么意思?