经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 移动开发 » iOS » 查看文章
【Objective-C】探索Category底层的实质
来源:cnblogs  作者:零零圈圈  时间:2019/7/15 8:53:39  对本文有异议

  无论一个类设计的多么完美,在未来的需求演进中,都有可能会碰到一些无法预测的情况。那怎么扩展已有的类呢?一般而言,继承和组合是不错的选择。但是在Objective-C 2.0中,又提供了category这个语言特性,可以动态地为已有类添加新行为。如今category已经遍布于Objective-C代码的各个角落,从Apple官方的framework到各个开源框架,从功能繁复的大型APP到简单的应用,catagory无处不在。本文对category做了比较全面的整理,希望对读者有所裨益。  

  Objective-C中类别特性的作用如下:

  (1)可以将类的实现分散到多个不同文件或多个不同框架中(补充新的方法)。

  (2)可以创建私有方法的前向引用。

  (3)可以向对象添加非正式协议。

  Objective-C中类别特性的局限性如下:

  (1)类别只能想原类中添加新的方法,且只能添加而不能删除或修改原方法,不能向原类中添加新的属性。

  (2)类别向原类中添加的方法是全局有效的而且优先级最高,如果和原类的方法重名,那么会无条件覆盖掉原来的方法。 

 一、Category的底层实现

   Objective-C 通过 Runtime 运行时来实现动态语言这个特性,所有的类和对象,在 Runtime 中都是用结构体来表示的,Category 在 Runtime 中是用结构体 category_t 来表示的,下面是结构体 category_t 具体表示:

  1. typedef struct category_t {
  2. const char *name;//类的名字 主类名字
  3. classref_t cls;//
  4. struct method_list_t *instanceMethods;//实例方法的列表
  5. struct method_list_t *classMethods;//类方法的列表
  6. struct protocol_list_t *protocols;//所有协议的列表
  7. struct property_list_t *instanceProperties;//添加的所有属性
  8. } category_t;

  从category的定义也可以看出category的可为(可以添加实例方法,类方法,甚至可以实现协议,添加属性)和不可为(无法添加实例变量)。

  我们将结合 runtime 的源码探究下 Category 的实现原理。打开 runtime 源码工程,在文件 objc-runtime-new.mm 中找到以下函数:

  1. void _read_images(header_info **hList, uint32_t hCount)
  2. {
  3. ...
  4. _free_internal(resolvedFutureClasses);
  5. }
  6. // Discover categories.
  7. for (EACH_HEADER) {
  8. category_t **catlist =
  9. _getObjc2CategoryList(hi, &count);
  10. for (i = 0; i < count; i++) {
  11. category_t *cat = catlist[i];
  12. Class cls = remapClass(cat->cls);
  13. if (!cls) {
  14. // Category's target class is missing (probably weak-linked).
  15. // Disavow any knowledge of this category.
  16. catlist[i] = nil;
  17. if (PrintConnecting) {
  18. _objc_inform("CLASS: IGNORING category \?\?\?(%s) %p with "
  19. "missing weak-linked target class",
  20. cat->name, cat);
  21. }
  22. continue;
  23. }
  24. // Process this category.
  25. // First, register the category with its target class.
  26. // Then, rebuild the class's method lists (etc) if
  27. // the class is realized.
  28. BOOL classExists = NO;
  29. if (cat->instanceMethods || cat->protocols
  30. || cat->instanceProperties)
  31. {
  32. addUnattachedCategoryForClass(cat, cls, hi);
  33. if (cls->isRealized()) {
  34. remethodizeClass(cls);
  35. classExists = YES;
  36. }
  37. if (PrintConnecting) {
  38. _objc_inform("CLASS: found category -%s(%s) %s",
  39. cls->nameForLogging(), cat->name,
  40. classExists ? "on existing class" : "");
  41. }
  42. }
  43. if (cat->classMethods || cat->protocols
  44. /* || cat->classProperties */)
  45. {
  46. addUnattachedCategoryForClass(cat, cls->ISA(), hi);
  47. if (cls->ISA()->isRealized()) {
  48. remethodizeClass(cls->ISA());
  49. }
  50. if (PrintConnecting) {
  51. _objc_inform("CLASS: found category +%s(%s)",
  52. cls->nameForLogging(), cat->name);
  53. }
  54. }
  55. }
  56. }
  57. // Category discovery MUST BE LAST to avoid potential races
  58. // when other threads call the new category code before
  59. // this thread finishes its fixups.
  60. // +load handled by prepare_load_methods()
  61. ...
  62. }

  我们可以知道在这个函数中对 Category 做了如下处理:

  (1)将 Category 和它的主类(或元类)注册到哈希表中;

  (2)如果主类(或元类)已实现,那么重建它的方法列表;

  Category的实现原理:

  • 在编译时期,会将分类中实现的方法生成一个结构体 method_list_t 、将声明的属性生成一个结构体 property_list_t ,然后通过这些结构体生成一个结构体 category_t
  • 然后将结构体 category_t 保存下来
  • 在运行时期,Runtime 会拿到编译时期我们保存下来的结构体 category_t
  • 然后将结构体 category_t 中的实例方法列表、协议列表、属性列表添加到主类中
  • 将结构体 category_t 中的类方法列表、协议列表添加到主类的 metaClass 中

二、为何Category中的方法优先级高于原类中的方法?

  category_t 中的方法列表是插入到主类的方法列表前面(类似利用链表中的 next 指针来进行插入),所以这里 Category 中实现的方法并不会真正的覆盖掉主类中的方法,只是将 Category 的方法插到方法列表的前面去了。运行时在查找方法的时候是顺着方法列表的顺序查找的,它只要一找到对应名字的方法,就会停止查找,这里就会出现覆盖方法的这种假象了。
  1. // 这里大概就类似这样子插入
  2. newproperties->next = cls->data()->properties;
  3. cls->data()->properties = newproperties;,

三、为何Category中不能添加实例变量?

  通过结构体 category_t ,我们就可以知道,在 Category 中我们可以增加实例方法、类方法、协议、属性。这里没有 objc_ivar_list 结构体,代表我们不可以在分类中添加实例变量。
  因为在运行期,对象的内存布局已经确定,如果添加实例变量就会破坏类的内部布局,这个就是 Category 中不能添加实例变量的根本原因。
 

原文链接:http://www.cnblogs.com/xjf125/p/11179570.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号