-
Notifications
You must be signed in to change notification settings - Fork 4
objc_class
objc_class的内存分布:
-
isa
:继承自objc_object的isa,占8字节 -
superclass
:objc_class类型的一个结构体指针,占8字节 -
cache
:方法缓存、获取占16字节 -
bits
:可以指向class_rw_t和class_ro_t,可以通过首地址平移32字节获取(isa + superclass + cache => 8 + 8 + 16)-
class_rw_t
:运行时生成,包含class_ro_t。插入分类的方法、协议、属性,不能添加成员变量,可读可写 -
class_ro_t
:编译期生成的,只读的,它存储了当前类在编译期就已经确定的属性、方法以及协议,它里面是没有分类中定义的方法和协议 ( 这里面的list 都是一维数组)
-
Class
类型其实就是指向objc_class(objc-runtime-new.h) 结构体的指针
/// An opaque type that represents an Objective-C class.
typedef struct objc_class *Class;
objc_class继承自objc_object
, 结构体里存储了父类、方法列表等信息
struct objc_class : objc_object {
// Class ISA;
Class superclass;
cache_t cache; // formerly cache pointer and vtable
class_data_bits_t bits; // class_rw_t * plus custom rr/alloc flags
class_rw_t *data() const {
return bits.data();
}
考虑篇幅等原因,其他静态成员变量、方法省略...
}
-
cache_t 定义
struct cache_t { #if CACHE_MASK_STORAGE == CACHE_MASK_STORAGE_OUTLINED explicit_atomic<struct bucket_t *> _buckets; // 8字节 explicit_atomic<mask_t> _mask; // 4字节 #elif CACHE_MASK_STORAGE == CACHE_MASK_STORAGE_HIGH_16 explicit_atomic<uintptr_t> _maskAndBuckets; mask_t _mask_unused; explicit_atomic<uintptr_t> _maskAndBuckets; mask_t _mask_unused; #if __LP64__ uint16_t _flags; // 2字节 #endif uint16_t _occupied; // 2字节 考虑篇幅等原因,其他静态成员变量、方法省略... }
cache_t的主要成员
-
_buckets
:struct bucket_t *
类型的一个结构体指针 -
_mask
:mask_t
类型, 在64位下为4字节 -
_flags
:uint16_t
类型, 2字节 -
_occupied
:uint16_t
类型, 2字节
-
cache_t
通过散列表来缓存曾经调用过的方法,可以提高方法的查找速度 (这里指的就是 msg_send 的快速查找)
通过objc_class
中的定义, 相当于为btis
提供了一个get
方法, 其返回值类型为class_rw_t
class_rw_t *data() const {
return bits.data();
}
bits.data()
源码
class_rw_t* data() const {
return (class_rw_t *)(bits & FAST_DATA_MASK);
}
将bits与FAST_DATA_MASK进行位运算,只取其中的3, 47位转换成class_rw_t*返回
class_rw_t的内存结构
class_rw_t源码
struct class_rw_t {
// Be warned that Symbolication knows the layout of this structure.
uint32_t flags;
uint16_t witness;
#if SUPPORT_INDEXED_ISA
uint16_t index;
#endif
explicit_atomic<uintptr_t> ro_or_rw_ext;
Class firstSubclass;
Class nextSiblingClass;
const method_array_t methods() const { //方法
auto v = get_ro_or_rwe();
if (v.is<class_rw_ext_t *>()) {
return v.get<class_rw_ext_t *>()->methods;
} else {
return method_array_t{v.get<const class_ro_t *>()->baseMethods()};
}
}
const property_array_t properties() const { //属性
auto v = get_ro_or_rwe();
if (v.is<class_rw_ext_t *>()) {
return v.get<class_rw_ext_t *>()->properties;
} else {
return property_array_t{v.get<const class_ro_t *>()->baseProperties};
}
}
const protocol_array_t protocols() const { //协议
auto v = get_ro_or_rwe();
if (v.is<class_rw_ext_t *>()) {
return v.get<class_rw_ext_t *>()->protocols;
} else {
return protocol_array_t{v.get<const class_ro_t *>()->baseProtocols};
}
}
...
}
从源码中,我们可以知道class_rw_t是一个结构体,存储了方法列表,属性列表,协议列表等等。
class_ro_t的内存结构
-
class_rw_t
是在运行时生成的,它在realizeClass
中生成,它包含了class_ro_t
。 它在_objc_init
方法中关于dyld
的回调的map_images
中最终将分类的方法与协议都插入到自己的方法列表、协议列表中。
它不包含成员变量列表,因为成员变量列表是在编译期就确定好的,它只保存在class_ro_t
中。不过class_rw_t
中包含了一个指向class_ro_t
的指针。 -
class_ro_t
存放的是编译期间就确定的;而class_rw_t
是在runtime
时才确定,它会先将class_ro_t
的内容拷贝过去,然后再将当前类的分类的这些属性、方法等拷贝到其中。所以可以说class_rw_t
是class_ro_t
的超集,当然实际访问类的方法、属性等也都是访问的class_rw_t
中的内容 -
class_rw_t
中的property_list
中只有属性,没有成员变量- 通过
{}
定义的成员变量,会存储在类的bits
属性中,通过bits -> data() ->ro() -> ivars
获取成员变量列表,除了包括成员变量,还包括属性定义的成员变量 - 通过
@property
定义的属性,也会存储在bits
属性中,通过bits -> data() -> properties() -> list
获取属性列表,其中只包含属性
- 通过
-
类中的方法列表除了包括实例方法,还包括属性的
set方法
和get方法
, 还有系统在底层添加了一个c++的.cxx_destruct
方法
ShenYj.github.io - 简书地址 - 返回首页
-
Apple
Common
蓝牙
LBS
音视频
- AVAudioSessionchange_route
- 切换线路
- StreamingKit
- Audio Unit 基础
OC 与 Swift 混编
Object-C
- 代码混淆
- autoreleasepool
- 忽略编译器(clang)警告
- 定时器
- 锁
- RunLoop
- block
- NS_REFINED_FOR_SWIFT
- NS_CLOSED_ENUM
- NS_TYPED_ENUM、NS_STRING_ENUM
- NS_TYPED_EXTENSIBLE_ENUM、NS_EXTENSIBLE_STRING_ENUM
- 关键字nonnull和nullable
- class、objc_getClass和object_getclass方法区别
- isKindOfClass和isMemberOfClass
- 应用程序的加载
- non-lazy classes & lazy classes
- load方法
- initialize方法
- 方法的本质
- 类型编码
- self和super
- 类的内存分析
Swift
- precondition
- 权限控制
- Array常用Api
- String初始化、定义
- String常用Api
- String截取演练
- Set定义、创建
- Set访问和修改
- Dictionary操作
- Dictionary和KeyValuePairs
- Dictionary与String转换
- 常用高阶函数
- enum原始值
- enum关联值
- enum遍历
- 递归enum
- enum内存分配
- 指针
- for循环
- break跳出循环
- 变量名与关键字冲突
- 类的定义
- 类的继承和初始化
- 关键字: final
- 关键字: mutating
- 关键字: lazy
- 修饰类方法的关键字: static
- 关键字: final、dynamic、objc和_dynamicReplacement
- 关键字:@dynamicMemberLookup和@dynamicCallable
- 关键字: propertyWrapper
- 自定义运算符
- 下标: subscript
- 扩展: extension
- 协议: protocol
- 协议和扩展
- 为什么需要泛型
- 泛型函数定义
- 泛型类型
- 泛型的类型约束
- 关联类型
- 为泛型定义要求
- 泛型下标
- 多线程
- Attributes
- 错误处理
- Codable
- DispatchSourceTimer
- Swift 5.x 演练: 更多功能编辑页
- Swift 5.x 类库收集
- 单元测试笔记
- 实例对象内存结构
- 元类型、Type、Self
- frozen
- convention
- Swift(5.3.2)源码编译
- SQLite.Swift类库演练
- Swift 5.5 关键字: async/await
- Swift 5.5 新特性: Continuations
- Swift 5.5 新特性: Actor
- Swift 方法调度
- Swift Mirror
- Swift 关键字: @_silgen_name
- Swift 关键字: @_disfavoredOverload
- swiftmodule
- Swift 5.6 新特性: Type placeholders
- Swift 5.6 新特性: #unavailable
- Swift 5.6 新特性: CodingKeyRepresentable
- Swift 5.6 新特性: existential any
- Swift 5.7 新特性: if-let/guard 语法简化
- Swift 5.7 新特性: Multi-statement closure type inference
- Swift 5.8 新特性: @backDeployed
- Swift 5.9 新特性: if switch expressions
- Swift 6.0 新特性:@preconcurrency
RxSwift
macOS - AppKit
-
iOS Assembly(ARM64)
-
C++
C++ 基础
- cout、cin
- 函数重载
- 默认参数
- extern "C"
- pragma once
- inline function
- const
- Reference
- 汇编
- 类和对象
- 堆空间内存管理
- Constructor
- Destructor
- 成员变量初始化
- 声明与实现分离
- namespace
- 继承
- 访问权限
- 初始化列表
- 多态:虚函数
- 多态:虚函数实现原理
- 多态:虚析构函数
- 多态:纯虚函数
- 多态:抽象类
- 多继承
- static
- static: 单例模式
- const 成员
- 引用类型成员
- 拷贝构造函数
- 调用父类的拷贝构造函数
- 浅拷贝、深拷贝
- 对象型参数和返回值
- 匿名对象
- 隐式构造
- 编译器自动生成的构造函数
- 友元
- 内部类
- 局部类
- 运算符重载
- 模板
- 类型转换
- C++标准
- Lambda
- 异常
- 智能指针
-
Flutter
Dart
Flutter
-
Go
Go 基础
-
Ruby
Ruby 基础
-
React-Native
React-Native
-
工具篇
-
Swift Package Manager
-
自动化
-
TroubleShooting
-
扩展