关于 Method Swizzling 这个东西,已经有很多高人写了详细的文章来介绍,我就不再班门弄斧,往深了说了。 而且不作延伸的话,这项技术本身也没有复杂到要长文论述的地步。 本文旨在帮助不熟悉这项技术的人,开始在实际开发过程中,尝试使用它。
比如说,在某个项目中,NSArray 实例的下面这个方法被调用了 N 多次
func containsObject(anObject: AnyObject) -> Bool
现在我想调试一下,看看如果这个方法返回 true,即数组包含我们传入的元素的时候,这个元素在数据的什么位置(index)。
func indexOfObject(anObject: AnyObject) -> Int
当然直接调用上面这个方法就可以知道 index,但是 <code>containsObject</code> 被使用了太多次,Xcode 现在又不支持 Swift 重构,懒得改了。那就写个新方法,给原方法加个可以输出 index 的功能,再用 swizz 替换一下两个方法的实现吧。
这里我贴了完整的一个 demo 的代码,你可以直接粘到 Xcode 里面运行。
import UIKit
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
let arr = NSArray(array: ["Swift","Method","Swizzling"])
print("-----Swizzling 之前-----")
print(arr.containsObject("Swift")) // true
NSArray.swizz() // 方法互换
print("-----Swizzling 之后-----")
print(arr.containsObject("Swift")) // 先输出 index,再 true
NSArray.swizz() // 方法再换回来
print("-----Swizzling 两下-----")
print(arr.containsObject("Swift")) // true
}
}
extension NSArray {
// 用来和默认方法进行替换的方法
func myContainsObject(anObject: AnyObject) -> Bool {
// 输出元素的 index,这是默认的原方法不具有的功能
if self.myContainsObject(anObject) {
print("index:\\(self.indexOfObject(anObject))")
}
// 不会产生死循环,因为运行期间,下面的方法已经被替换成了默认的 containsObject
return self.myContainsObject(anObject)
}
// 用来给不同方法互相替换的方法
class func swizz() {
let originalMethod = class_getInstanceMethod(NSArray.self, #selector(containsObject(_:)))
let swizzledMethod = class_getInstanceMethod(NSArray.self, #selector(myContainsObject(_:)))
method_exchangeImplementations(originalMethod, swizzledMethod)
}
}
console
【注意几点】