4

我正在关注Swift 和 Objective-C Runtime,它适用于普通方法。

我喜欢swizzle init 方法,据我了解,init 就像一个类方法。所以我尝试将 init 作为实例和类方法。但这似乎不起作用

我可以使用 Objective C 让它工作,只是想知道如何让它在 Swift 中工作

摘自我的要点

dispatch_once(&Static.token) {
            let originalSelector = Selector("init:source:destination:")
            let swizzledSelector = Selector("ftg_init:source:destination:")

            let originalMethod = class_getClassMethod(self, originalSelector)
            let swizzledMethod = class_getClassMethod(self, swizzledSelector)

            let didAddMethod = class_addMethod(self, originalSelector, method_getImplementation(swizzledMethod), method_getTypeEncoding(swizzledMethod))

            if didAddMethod {
                class_replaceMethod(self, swizzledSelector, method_getImplementation(originalMethod), method_getTypeEncoding(originalMethod))
            } else {
                method_exchangeImplementations(originalMethod, swizzledMethod);
            }
        }
4

2 回答 2

6

在为方法创建选择器时,您应该将其基于 Obj C 方法签名,因为 swizzling 是使用 Obj C 运行时完成的。

所以原来的选择器应该是

initWithIdentifier:source:destination:

现在这有点奇怪,因为您的 init 方法已定义为需要第一个参数上的标签(通过identifier两次),您要使用的选择器实际上是

ftg_initWithIdentifier:source:destination:

我能找到的唯一关于它的文档是关于从 Obj C 到 Swift 的翻译,但看起来从 Swift 到 Obj C 的情况正好相反。

接下来,init...是一个实例方法,因此您需要进行两项更改。您需要更改class_getClassMethodclass_getInstanceMethod并且需要class从您的ft_init...方法中删除。

所以当一切都说完了,你的代码应该是这样的(这对我有用)

dispatch_once(&Static.token) {
    let originalSelector = Selector("initWithIdentifier:source:destination:")
    let swizzledSelector = Selector("ftg_initWithIdentifier:source:destination:")

    let originalMethod = class_getInstanceMethod(self, originalSelector)
    let swizzledMethod = class_getInstanceMethod(self, swizzledSelector)

    let didAddMethod = class_addMethod(self, originalSelector, method_getImplementation(swizzledMethod), method_getTypeEncoding(swizzledMethod))

    if didAddMethod {
        class_replaceMethod(self, swizzledSelector, method_getImplementation(originalMethod), method_getTypeEncoding(originalMethod))
    } else {
        method_exchangeImplementations(originalMethod, swizzledMethod);
    }
}

func ftg_init(identifier identifier: String!,
    source: UIViewController,
    destination: UIViewController) -> UIStoryboardSegue {

    return ftg_init(identifier: identifier,
        source: source,
        destination: destination.ftg_resolve())
}
于 2015-12-16T22:50:59.593 回答
4

这是在 Swift 4.2 中为 UIImage 初始化程序执行此操作的方法:

@objc static func ftg_imageNamed(_ name: String) -> UIImage? {
    ...
}

private static func swizzleInitImplementation() {
    let originalSelector = #selector(UIImage.init(named:))
    let swizzledSelector = #selector(UIImage.ftg_imageNamed(_:))

    guard let originalMethod = class_getClassMethod(self, originalSelector), let swizzledMethod = class_getClassMethod(self, swizzledSelector) else {
        assertionFailure("The methods are not found!")
        return
    }

    let didAddMethod = class_addMethod(self, originalSelector, method_getImplementation(swizzledMethod), method_getTypeEncoding(swizzledMethod))
    if didAddMethod {
        class_replaceMethod(self, swizzledSelector, method_getImplementation(originalMethod), method_getTypeEncoding(originalMethod))
    } else {
        method_exchangeImplementations(originalMethod, swizzledMethod);
    }
}
于 2018-12-18T14:59:07.940 回答