objective-c - Swift isa 指针重映射或其他支持的方法调配

标签 objective-c swift objective-c-runtime

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/

相关文章:

ios - 添加 subview 不起作用

快速函数迭代可能反转的数组

objective-c - 了解 Objective-C 运行时

objective-c - 获取调用方法的对象

objective-c - 异步获取大量资源并将其保存到数据库 "asynchronously"使用哪种好的模式? (AFNetworking,核心数据)

objective-c - RestKit 0.20 — 核心数据 : error: Failed to call designated initializer on NSManagedObject class

objective-c - Cocoa:获取和更改第三方应用程序中的文本和选择(无需模拟按键或使用可访问性 API)

Objective-c 枚举对象到字符串

ios - 在 Swift 应用程序中实现 iAd

ios - 将对象转换(或复制)到 Objective-C 中的子类实例