Objective-C 中的 isa 不再是简单的结构体指针

embedded/2024/10/10 15:56:57/

isa__1">了解 Objective-C 中的 isa 指针内存结构

在 Objective-C 中,isa 指针是对象和类之间的重要桥梁。它不仅帮助运行时系统识别对象的类型,还参与了一些内存和性能优化。本文将深入讲解 isa 指针的内存结构,包括其在早期和现代实现中的演变。

isa__5">什么是 isa 指针?

每个 Objective-C 对象都有一个 isa 指针,它指向对象的类对象。类对象本身也是一个对象,它的 isa 指针指向一个元类对象(meta-class)。元类对象存储类方法,并且其 isa 指针最终指向根元类(通常是 NSObject 的元类)。

isa__9">早期的 isa 指针结构

在早期的 Objective-C 实现中,isa 指针简单地指向类对象的结构体。以下是一个典型的早期实现示例:

struct objc_object {Class isa; // 指向类对象的指针
};typedef struct objc_class *Class; // Class 的本质是 objc_class 类型的结构体指针
struct objc_class {Class isa; // 指向元类对象的指针Class super_class; // 指向父类对象的指针// 其他类相关的元数据
};

在这种结构下:

  • 对象的 isa 指针指向类对象。
  • 类对象的 isa 指针指向元类对象。
  • 元类对象的 isa 指针指向根元类对象。

isa__31">现代 isa 指针结构

在 64 位系统和现代 Objective-C 运行时中,isa 指针被重新设计为一个更复杂的联合体(union isa_t),它不仅包含指向类对象的指针,还包含其他标志位和信息,以优化内存使用和性能。以下是 isa_t 结构的一个简化示例:

union isa_t {isa_t() { }isa_t(uintptr_t value) : bits(value) { }Class cls; // 指向类对象的指针uintptr_t bits; // 包含位域信息的位模式struct {uintptr_t nonpointer        : 1;  // 是否启用优化的 non-pointer isauintptr_t has_assoc         : 1;  // 是否有关联对象uintptr_t has_cxx_dtor      : 1;  // 是否有 C++ 析构函数uintptr_t shiftcls          : 33; // 类指针(经过位移和压缩)uintptr_t magic             : 6;  // 调试用的魔数uintptr_t weakly_referenced : 1;  // 是否被弱引用uintptr_t deallocating      : 1;  // 是否正在释放uintptr_t has_sidetable_rc  : 1;  // 是否有辅助引用计数表uintptr_t extra_rc          : 19; // 额外的引用计数};
};

结构字段解释

  • nonpointer:指示 isa 是否为非指针类型(优化内存布局,存储额外信息)。
  • has_assoc:对象是否有关联引用(Associative References)。
  • has_cxx_dtor:对象是否有 C++ 析构函数,需要调用析构函数。
  • shiftcls:类指针,存储对象的类信息(经过位移和压缩)。
  • magic:用于调试和运行时验证的魔数(magic number)。
  • weakly_referenced:对象是否被弱引用指向。
  • deallocating:对象是否正在被释放。
  • has_sidetable_rc:对象的引用计数是否存储在辅助表(Side Table)中。
  • extra_rc:额外的引用计数,用于优化内存占用。

引用计数的存储与管理

在早期的 Objective-C 实现中,引用计数通常作为对象结构的一部分直接存储在对象中。例如:

struct objc_object {Class isa; // 指向类对象的指针uintptr_t retainCount; // 引用计数
};

在现代的 Objective-C 运行时中,引用计数通过 isa 指针的优化结构和 Side Table 辅助数据结构进行管理。

  • Inline Reference Counting:部分引用计数信息被存储在 isa 指针的优化结构中,例如 extra_rc 字段。
  • Side Table:当引用计数超出 isa 指针所能表示的范围时,引用计数会存储在一个称为 Side Table 的辅助数据结构中。

isa__85">Modern isa 指针的优势

  • 内存优化:通过将更多信息(如引用计数、标志位)存储在 isa 指针中,减少了对其他内存区域的访问,提升了性能。
  • 性能提升:减少了内存读取操作,因为可以在一次内存读取中获取更多信息。
  • 更丰富的元数据:可以包含更多运行时信息,有助于提高运行时的灵活性和效率。

使用示例

虽然开发者在日常编码中通常不直接与 isa 指针交互,但理解其结构对于调试和优化性能是有帮助的。以下是一个使用示例,通过访问对象的类信息来显示对象的类型:

#import <Foundation/Foundation.h>
#import <objc/runtime.h>@interface MyClass : NSObject
@end@implementation MyClass
@endint main(int argc, const char * argv[]) {@autoreleasepool {MyClass *obj = [[MyClass alloc] init];Class cls = object_getClass(obj);NSLog(@"Class name: %s", class_getName(cls));// 访问 isa 指针信息(需要通过运行时函数)NSLog(@"isa pointer: %p", *(uintptr_t *)obj);}return 0;
}

总结

isa 指针在 Objective-C 运行时中扮演着重要角色,从早期简单的指向类对象,到现代复杂的 isa_t 结构,它帮助优化了内存使用和性能。理解 isa 指针的演变和内存结构,可以帮助我们更好地掌握 Objective-C 的运行时机制,并编写高效的代码。

希望这篇文章能帮助你深入了解 Objective-C 中 isa 指针的内存结构。如有任何问题或建议,欢迎留言讨论。


http://www.ppmy.cn/embedded/59035.html

相关文章

【架构】分布式与微服务架构解析

分布式与微服务架构解析 一、分布式1、什么是分布式架构2、为什么需要分布式架构3、分布式架构有哪些优势&#xff1f;4、分布式架构有什么劣势&#xff1f;5、分布式架构有哪些关键技术&#xff1f;6、基于分布式架构如何提高其高性能&#xff1f;7、如何基于架构提高系统的稳…

Redis基础教程(二十):Java使用Redis

&#x1f49d;&#x1f49d;&#x1f49d;首先&#xff0c;欢迎各位来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里不仅可以有所收获&#xff0c;同时也能感受到一份轻松欢乐的氛围&#xff0c;祝你生活愉快&#xff01; &#x1f49d;&#x1f49…

Chromium CI/CD 之Jenkins实用指南2024-Windows安装篇(一)

1. 引言 在现代软件开发过程中&#xff0c;持续集成和持续部署&#xff08;CI/CD&#xff09;是确保高效、稳定软件交付的关键实践。Jenkins作为一款广泛使用的自动化服务器&#xff0c;通过其强大的插件体系和灵活的配置&#xff0c;支持各种操作系统和开发环境。为了帮助开发…

(五)共享模型之无锁

问题提出 有如下需求&#xff0c;保证 account.withdraw 取款方法的线程安全 interface Account {// 获取余额Integer getBalance();// 取款void withdraw(Integer amount);static void demo(Account account) {List<Thread> ts new ArrayList<>();long start …

树莓派编译安装php出现cc: fatal error: Killed signal terminated program cc1

树莓派编译安装php出现cc: fatal error: Killed signal terminated program cc1 dd if/dev/zero of/swapfile bs1k count2048000 mkswap /swapfile swapon /swapfile swapon -s echo "/var/swapfile swap swap defaults 0 0" >> /etc/fstab swapoff /swapfile…

使用sklearn的基本流程

scikit-learn&#xff0c;通常简称为 sklearn&#xff0c;是一个开源的Python库&#xff0c;是基于 Python 编程语言的一个非常流行的机器学习库。它建立在 NumPy 和 SciPy 这两个科学计算库之上&#xff0c;并与 Matplotlib 配合使用&#xff0c;为数据预处理、模型训练、评估…

Docker修改Postgresql密码

在Docker环境中&#xff0c;对已运行的PostgreSQL数据库实例进行密码更改是一项常见的维护操作。下面将详述如何通过一系列命令行操作来实现这一目标。 修改方式 查看容器状态及信息 我们需要定位到正在运行的PostgreSQL容器以获取其相关信息。执行以下命令列出所有正在运行…

Python 3.x 下的 3D 游戏引擎

在 Python 3.x 中&#xff0c;有几个比较流行的用于开发 3D 游戏的引擎和库。虽然 Python 自身不是一个主流的游戏开发语言&#xff0c;但是可以通过这些库和引擎结合其它语言或者底层渲染引擎来实现复杂的游戏开发。 1、问题背景 在 Linux 系统中&#xff0c;尤其是 Debian 7…