iOS - Runtime - Class的结构

文章目录

  • [iOS - Runtime - Class的结构](#iOS - Runtime - Class的结构)
    • 前言
    • [1. Class的结构](#1. Class的结构)
      • [1.1 Class的结构](#1.1 Class的结构)
        • [1.1.1 objc_class](#1.1.1 objc_class)
        • [1.1.2 class_rw_t](#1.1.2 class_rw_t)
        • [1.1.3 class_ro_t](#1.1.3 class_ro_t)
      • [1.2 class_rw_t和class_ro_t的区别](#1.2 class_rw_t和class_ro_t的区别)
      • [1.3 class_rw_t和class_ro_t的关系](#1.3 class_rw_t和class_ro_t的关系)
        • [1.3.1 分析关系](#1.3.1 分析关系)
        • [1.3.2 原因](#1.3.2 原因)
      • [1.4 method_t](#1.4 method_t)
        • [1.4.1 Type Encoding](#1.4.1 Type Encoding)
        • [1.4.2 types说明](#1.4.2 types说明)
          • [1.4.2.1 - (void)test](#1.4.2.1 - (void)test)
          • [1.4.2.2 - (void)test:(NSString *)name](#1.4.2.2 - (void)test:(NSString *)name)

iOS - Runtime - Class的结构

前言

1. Class的结构

前面我们学习OC对象的时候,了解了class类对象、meta-class元类对象

实际上,class类对象、meta-class元类对象底层结构体都是class元类对象可以认为是特殊的类对象。他们底层结构是一样的,只不过存储的内容有所区别

1.1 Class的结构

Class的结构如下图所示:

1.1.1 objc_class

里面存储了

  • isa
  • superclass
  • cache
  • bits

其中的bits& FAST_DATA_MASK位运算之后,拿到class_rw_tclass_rw_t里面的ro则是class_ro_t

1.1.2 class_rw_t
  • class_rw_t里面的methodspropertiesprotocols是二维数组,是可读可写的,包含了类的初始内容分类的内容
1.1.3 class_ro_t
  • class_ro_t里面的baseMethodListbaseProtocolsivarsbaseProperties是一维数组,是只读的,包含了类的初始内容

1.2 class_rw_t和class_ro_t的区别

对于他们里面的主要内容:

  1. class_rw_tmethodspropertiesprotocols
  2. class_ro_tbaseMethodListbaseProtocolsivarsbaseProperties

区别主要是:

  • class_rw_t的主要成员是二维数组class_ro_t的主要成员是一维数组
  • class_rw_t的主要成员是可读可写的,class_ro_t是只读的

1.3 class_rw_t和class_ro_t的关系

class_rw_tclass_ro_t里面的结构很像,但是class_ro_t的成员基本上带了base

1.3.1 分析关系

其实刚开始的时候不存在rw的,class里面的东西都保存在ro里面,bits取出来的其实是rorw是后面才生成的

因为rw是可读可写的,生成的时候,会把ro里面的东西合并进去,然后把rw设置为datacls->setData(rw),此时rw就替换了原本ro的位置,同时rw把ro属性指向原来的ro对象

1.3.2 原因

class_rw_t里面的数据基本上是二维数组,并且是可读可写。

我们知道分类的东西是runtime阶段才附加进去的,在将分类的东西合并到class里面的时候。

因此他们之间是相互配合的,class_ro_t先是存储了编译期间固定的东西,运行时一些动态的东西则是存入class_rw_tclass_rw_t在初始化前会将class_ro_t的东西先附加进去。于是class_rw_t里面就有完整的东西

1.4 method_t

  • method_t是对方法\ 函数的封装
  • IMP代表函数的具体实现
  • SEL代表方法\函数名,一般叫做选择器,底层结构跟char *类似
    1. 可以通过@selector()sel_registerName()获得
    2. 可以通过sel_getName()NSStringFromSelector()转成字符串
    3. 不同类相同名字的方法,所对应的方法选择器相同
  • types包含了函数返回值、参数编码的字符串
返回值 参数1 参数2 ... 参数n
1.4.1 Type Encoding

iOS中提供了一个叫做@encode的指令,可以将具体的类型表示成字符串编码

1.4.2 types说明

使用Xcode 12.5.1 iPhone(12.4)模拟器

1.4.2.1 - (void)test

方法- (void)test;对应的typesv16@0:8

  • - (void)test;实际上C语言函数为:- (void)test:(id)self _cmd:(SEL)_cmd,包含self_cmd两个参数

于是types表示如下表:

v 16 @ 0 : 8
void 无返回值 2个参数共16字节 1个参数:id类型 1个参数:从0字节开始 2个参数:SEL类型 2个参数:从8字节开始
1.4.2.2 - (void)test:(NSString *)name

test方法增加一个name参数,此时typesv24@0:8@16

types表示如下表:

于是types表示如下表:

v 24 @ 0 : 8 @ 16
void 无返回值 3个参数共24字节 1个参数:id类型 1个参数:从0字节开始 2个参数:SEL类型 2个参数:从8字节开始 3个参数:id类型 3个参数:从16字节开始

@oubijiexi

相关推荐
未来侦察班9 小时前
一晃13年过去了,苹果的Airdrop依然很坚挺。
macos·ios·苹果vision pro
锐意无限15 小时前
Swift 扩展归纳--- UIView
开发语言·ios·swift
符哥200815 小时前
用Apollo + RxSwift + RxCocoa搭建一套网络请求框架
网络·ios·rxswift
文件夹__iOS19 小时前
AsyncStream 进阶实战:SwiftUI 全局消息流极简实现
ios·swiftui·swift
2501_916008891 天前
深入解析iOS机审4.3原理与混淆实战方法
android·java·开发语言·ios·小程序·uni-app·iphone
忆江南1 天前
Flutter深度全解析
ios
山水域1 天前
Swift 6 严格并发检查:@Sendable 与 Actor 隔离的深度解析
ios
楚轩努力变强1 天前
iOS 自动化环境配置指南 (Appium + WebDriverAgent)
javascript·学习·macos·ios·appium·自动化
游戏开发爱好者82 天前
日常开发与测试的 App 测试方法、查看设备状态、实时日志、应用数据
android·ios·小程序·https·uni-app·iphone·webview
黑码哥2 天前
ViewHolder设计模式深度剖析:iOS开发者掌握Android列表性能优化的实战指南
android·ios·性能优化·跨平台开发·viewholder