【问题标题】:How to implement method swizzling swift 3.0?如何实现方法 swizzling swift 3.0?
【发布时间】:2017-01-26 12:52:12
【问题描述】:

如何在 Swift 3.0 中实现方法调配?

我已经阅读了nshipster article 的相关信息,但在这段代码中

struct Static {
    static var token: dispatch_once_t = 0
}

编译器给我一个错误

dispatch_once_t 在 Swift 中不可用:使用延迟初始化 改为全局变量

【问题讨论】:

  • 错误信息有点迟钝;可以使用static 变量,因为static 变量只自动创建一次,这发生在它们第一次被访问时(即懒惰地)。 static var token = 0 是在 Swift 3 中执行此操作的正确方法。Source

标签: ios swift xcode swift3 swizzling


【解决方案1】:

首先dispatch_once_t 在 Swift 3.0 中不可用。 您可以从两种选择中进行选择:

  1. 全局变量

  2. structenumclass 的静态属性

更多详情请看Whither dispatch_once in Swift 3

为了不同的目的,你必须使用不同的 swizzling 实现

  • Swizzling CocoaTouch 类,例如 UIViewController;
  • Swizzling 自定义 Swift 类;

混合 CocoaTouch 类

使用全局变量对UIViewControllerviewWillAppear(_:) 进行调配示例

private let swizzling: (UIViewController.Type) -> () = { viewController in

    let originalSelector = #selector(viewController.viewWillAppear(_:))
    let swizzledSelector = #selector(viewController.proj_viewWillAppear(animated:))

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

    method_exchangeImplementations(originalMethod, swizzledMethod) }

extension UIViewController {

    open override class func initialize() {
        // make sure this isn't a subclass
        guard self === UIViewController.self else { return }
        swizzling(self)
    }

    // MARK: - Method Swizzling

    func proj_viewWillAppear(animated: Bool) {
        self.proj_viewWillAppear(animated: animated)

        let viewControllerName = NSStringFromClass(type(of: self))
        print("viewWillAppear: \(viewControllerName)")
    } 
 }

Swizzling 自定义 Swift 类

要在您的 Swift 类中使用方法调配,您必须遵守两个要求 (for more details):

  • 包含要混合的方法的类必须扩展 NSObject
  • 您要调配的方法必须具有dynamic 属性

以及自定义Swift基类Person的示例swizzling方法@

class Person: NSObject {
    var name = "Person"
    dynamic func foo(_ bar: Bool) {
        print("Person.foo")
    }
}

class Programmer: Person {
    override func foo(_ bar: Bool) {
        super.foo(bar)
        print("Programmer.foo")
    }
}

private let swizzling: (Person.Type) -> () = { person in

    let originalSelector = #selector(person.foo(_:))
    let swizzledSelector = #selector(person.proj_foo(_:))

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

    method_exchangeImplementations(originalMethod, swizzledMethod)
}

extension Person {

    open override class func initialize() {
        // make sure this isn't a subclass
        guard self === Person.self else { return }
        swizzling(self)
    }

    // MARK: - Method Swizzling

    func proj_foo(_ bar: Bool) {
        self.proj_foo(bar)

        let className = NSStringFromClass(type(of: self))
        print("class: \(className)")
    }
}

【讨论】:

  • 在这种情况下,class_addMethod() 是否会返回 true?因为 swift #selector 需要关联的方法存在,所以originalSelector 会一直存在,导致class_addMethod “失败”。我发现只包含对method_exchangeImplementations() 的调用是安全的,而忽略对classAddMethod()class_replaceMethod() 的调用。
  • 在未来的 Swift 版本中将不允许使用 ˆinitialize()` 方法。 Swift 3.1 更新日志: 现在,当 NSObject 子类尝试覆盖类初始化方法时,Swift 会发出警告。如果没有其他副作用,Swift 不保证对类名的引用会触发 Objective-C 类的实现,这会导致 Swift 代码尝试覆盖初始化时出现错误。
  • @Alessandro 对如何在这些限制条件下执行 swizzling 有什么建议吗?
  • 凹凸。关于下一个进行调酒的最佳地点有什么想法吗?
  • 您可以在应用程序委托中添加一个方法,以便在调用 application(_:didFinishLaunchingWithOptions:) 时执行。检查Kickstarter App repo中的AppDelegate;特别是 UIViewController.doBadSwizzleStuff() 实现。 @ilan
【解决方案2】:

@TikhonovAlexander:很好的答案

我修改了调酒器以采用两个选择器并使其更通用。

斯威夫特 4 / 斯威夫特 5

private let swizzling: (AnyClass, Selector, Selector) -> () = { forClass, originalSelector, swizzledSelector in
    guard
        let originalMethod = class_getInstanceMethod(forClass, originalSelector),
        let swizzledMethod = class_getInstanceMethod(forClass, swizzledSelector)
    else { return }
    method_exchangeImplementations(originalMethod, swizzledMethod)
}

extension UIView {
    
    static let classInit: Void = {            
        let originalSelector = #selector(layoutSubviews)
        let swizzledSelector = #selector(swizzled_layoutSubviews)
        swizzling(UIView.self, originalSelector, swizzledSelector)
    }()
    
    @objc func swizzled_layoutSubviews() {
        swizzled_layoutSubviews()
        print("swizzled_layoutSubviews")
    }
    
}

// perform swizzling in AppDelegate.init()

@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {

    override init() {
        super.init()
        UIView.classInit
    }

}

斯威夫特 3

private let swizzling: (AnyClass, Selector, Selector) -> () = { forClass, originalSelector, swizzledSelector in
    let originalMethod = class_getInstanceMethod(forClass, originalSelector)
    let swizzledMethod = class_getInstanceMethod(forClass, swizzledSelector)
    method_exchangeImplementations(originalMethod, swizzledMethod)
}

// perform swizzling in initialize()

extension UIView {
    
    open override class func initialize() {
        // make sure this isn't a subclass
        guard self === UIView.self else { return }
        
        let originalSelector = #selector(layoutSubviews)
        let swizzledSelector = #selector(swizzled_layoutSubviews)
        swizzling(self, originalSelector, swizzledSelector)
    }
    
    func swizzled_layoutSubviews() {
        swizzled_layoutSubviews()
        print("swizzled_layoutSubviews")
    }
    
}

【讨论】:

  • 我尝试使用本文中提到的 swift 4 代码来调整 UIColor 描述选择器。 let originalSelector = #selector(description) let swizzledSelector = #selector(colorDescription) swizzling(UIColor.self, originalSelector, swizzledSelector) 但它不起作用。我不得不使用let instance: UIColor = UIColor.red let aClass: AnyClass! = object_getClass(instance) 而不是UIColor.self。为什么会这样?
  • 由于此处详述的原因,这不是一种混合方法的安全方法:blog.newrelic.com/2014/04/16/right-way-to-swizzle
  • 你为什么使用块而不是方法?因为你只是复制了一些你不明白的东西。之所以使用这些块,是因为在博客文章中解释了原因……你是如何实现它的,你只是绕着它走,让它没用。不要因为有人在不理解他们为什么这样做的情况下做了某件事而采用了它。
  • @hashier,你为什么要与块或方法争论?据我了解,使用 classInit 块很好,它可以保证 classInit 只能执行一次。你能把你提到的帖子链接给我吗?
  • Swift 4 无法编译。您需要将@objc 添加到swizzled_layoutSubviews,并对swizzling 内部的方法进行可选检查。
【解决方案3】:

在 Playground 中嬉戏
Swift 5

import Foundation
class TestSwizzling : NSObject {
    @objc dynamic func methodOne()->Int{
        return 1
    }
}

extension TestSwizzling {
    @objc func methodTwo()->Int{
        // It will not be a recursive call anymore after the swizzling
        return 2
    }
    //In Objective-C you'd perform the swizzling in load(),
    //but this method is not permitted in Swift
    func swizzle(){
        let i: () -> () = {
            let originalSelector = #selector(TestSwizzling.methodOne)
            let swizzledSelector = #selector(TestSwizzling.methodTwo)
            let originalMethod = class_getInstanceMethod(TestSwizzling.self, originalSelector);
            let swizzledMethod = class_getInstanceMethod(TestSwizzling.self, swizzledSelector)
            method_exchangeImplementations(originalMethod!, swizzledMethod!)
            print("swizzled!")
        }
        i()
    }
}

var c = TestSwizzling()
print([c.methodOne(), c.methodTwo()])  // [1, 2]
c.swizzle()                            // swizzled!
print([c.methodOne(), c.methodTwo()])  // [2, 1]

【讨论】:

  • 结果打印([c.methodOne(), c.methodTwo()]) ==> // [1, 2].因为您声明 methodTwo() 返回 2。请更新您的代码以避免误导。
【解决方案4】:

Swiftswizzling

@objcMembers
class sA {
    dynamic
    class func sClassFooA() -> String {
        return "class fooA"
    }

    dynamic
    func sFooA() -> String {
        return "fooA"
    }
}

@objcMembers
class sB {
    dynamic
    class func sClassFooB() -> String {
        return "class fooB"
    }

    dynamic
    func sFooB() -> String {
        return "fooB"
    }
}

Swizzling.swift

import Foundation

@objcMembers
public class Swizzling: NSObject {

    public class func sExchangeClass(cls1: AnyClass, sel1: Selector, cls2: AnyClass, sel2: Selector) {

        let originalMethod = class_getClassMethod(cls1, sel1)
        let swizzledMethod = class_getClassMethod(cls2, sel2)

        method_exchangeImplementations(originalMethod!, swizzledMethod!)
    }

    public class func sExchangeInstance(cls1: AnyClass, sel1: Selector, cls2: AnyClass, sel2: Selector) {

        let originalMethod = class_getInstanceMethod(cls1, sel1)
        let swizzledMethod = class_getInstanceMethod(cls2, sel2)

        method_exchangeImplementations(originalMethod!, swizzledMethod!)
    }
}

通过 Swift 使用

func testSExchangeClass() {
    Swizzling.sExchangeClass(cls1: sA.self, sel1: #selector(sA.sClassFooA), cls2: sB.self, sel2: #selector(sB.sClassFooB))

    XCTAssertEqual("class fooB", sA.sClassFooA())
}

func testSExchangeInstance() {
    Swizzling.sExchangeInstance(cls1: sA.self, sel1: #selector(sA.sFooA), cls2: sB.self, sel2: #selector(sB.sFooB))

    XCTAssertEqual("fooB", sA().sFooA())
}

[Add Objective-C as an consumer]

通过 Objective-C 使用

- (void)testSExchangeClass {
    [Swizzling sExchangeClassWithCls1:[cA class] sel1:@selector(cClassFooA) cls2:[cB class] sel2:@selector(cClassFooB)];

    XCTAssertEqual(@"class fooB", [cA cClassFooA]);
}

- (void)testSExchangeInstance {
    [Swizzling sExchangeInstanceWithCls1:[cA class] sel1:@selector(cFooA) cls2:[cB class] sel2:@selector(cFooB)];

    XCTAssertEqual(@"fooB", [[[cA alloc] init] cFooA]);
}

[Objective-C swizzling]

【讨论】:

    【解决方案5】:

    斯威夫特 5.1

    Swift 使用 Objective-C 运行时特性来进行方法混用。这里有两种方法。

    注意:open override class func initialize() {} 不再允许使用。

    1. 类继承NSObject,方法必须有dynamic属性

      class Father: NSObject {
         @objc dynamic func makeMoney() {
             print("make money")
         }
      }
      extension Father {
         static func swizzle() {
             let originSelector = #selector(Father.makeMoney)
             let swizzleSelector = #selector(Father.swizzle_makeMoney)
             let originMethod = class_getInstanceMethod(Father.self, originSelector)
             let swizzleMethod = class_getInstanceMethod(Father.self, swizzleSelector)
             method_exchangeImplementations(originMethod!, swizzleMethod!)
         }
         @objc func swizzle_makeMoney() {
             print("have a rest and make money")
         }
      }
      Father.swizzle()
      var tmp = Father()
      tmp.makeMoney() //  have a rest and make money
      tmp.swizzle_makeMoney() // make money
      
      1. 使用@_dynamicReplacement(for: )

         class Father {
             dynamic func makeMoney() {
                 print("make money")
             }
         }
         extension Father {
             @_dynamicReplacement(for: makeMoney())
             func swizzle_makeMoney() {
                 print("have a rest and make money")
             }
         }
         Father().makeMoney() // have a rest and make money
      

    【讨论】:

    • 如果我在 Pod 中使用 @_dynamicReplacement (for:),当我在模拟器中运行它时,我会得到一个错误:“不支持使用减法表达式重定位,符号 '' 不能在减法表达式”。有没有办法解决或避免?
    • 我问过这个question
    【解决方案6】:

    试试这个框架:https://github.com/623637646/SwiftHook

    let object = MyObject()
    let token = try? hookBefore(object: object, selector: #selector(MyObject.noArgsNoReturnFunc)) {
        // run your code
        print("hooked!")
    }
    object.noArgsNoReturnFunc()
    token?.cancelHook() // cancel the hook
    

    在 Swift 中挂钩方法非常容易。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2018-12-17
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多