Swift 类有类似 isa 指针的东西可以重新映射吗?
我们已经看到 Swift uses a more static method dispatch与 objective-C 相比,后者(除非来自 Foundation/NSObject 的类服务)防止在运行时基于重新映射方法实现的 swizzling 风格。
我想知道我们将如何实现基于方法拦截的动态特性,例如观察者模式、通知等?目前所有这些东西都是由 Objective-C 层提供的,并且可以很容易地集成到 Swift 中。但是,如果我们想在自己的框架(或应用程序)中提供这些功能,是否有必要在 Objective-C 中实现它们?我会假设有一种方法可以“本地”进行。
Objective-C 常见的另一种调配是重新映射 isa 指针以动态生成子类。 Swift 支持这种调配吗?如果不是,拦截任意方法调用的支持方式是什么?
编辑: 正如@jatoben 指出的那样,从 arm64 开始,isa 重映射必须通过调用 object_setClass() 而不是通过直接访问值来完成。这仍然被称为“isa 指针调配”
最佳答案
看起来方法交换和 isa 指针重映射技术只有在 Swift 类将 NSObject 作为父类(super class)(直接或更上层)时才有效。当 Swift 类没有父类(super class)或其他一些非 Foundation 基类时,它目前不起作用。
下面的测试表明了这一点:
类别:小鸟
class Birdy: NSObject {
func sayHello()
{
print("tweet tweet")
}
}
类别:HodorBirdy
class HodorBirdy: Birdy {
override func sayHello()
{
super.sayHello()
print("hodor hodor")
}
}
测试:
func testExample() {
let birdy : Birdy = Birdy()
object_setClass(birdy, HodorBirdy.self)
birdy.sayHello();
}
并且输出符合预期:
tweet tweet
hodor hodor
在这个测试中,基类和子类都是预先创建的。尽管它们也可以使用 Objective-C 运行时动态创建,只要该类具有 NSObject 作为祖先即可。
当 Swift 类不是从 Objective-C 基础派生时,编译器将支持基于静态或 vtable 的分派(dispatch),因此在这种情况下方法拦截将如何工作根本不清楚!
除非语言/编译器特别允许,否则我们将放弃动态以支持性能。 (拦截是“动态”行为的基础,可以在编译时或运行时完成。在没有虚拟机的静态或 vtable 调度的情况下,仅适用于编译时)。
关于objective-c - Swift isa 指针重映射或其他支持的方法调配,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24050500/