iOS主要知识点梳理回顾-5-运行时方法交换

方法交换可以放在 +load+initialize 方法中,也可以自己根据时机来空,比如开启某个开关后才需要交换方法。如果是在+load中调用,交换工作会在类加载时(程序启动)自动调用;如果是在+initialize中调用,则会在该类初次发送消息的时候调用(如初始化、调用类方法等)。我们可以根据实际需要决定调用调用时机

实现交换

objectivec 复制代码
#import <objc/runtime.h>

@implementation UIViewController (Swizzling)

+ (void)load {
    // 获取原始的方法
    Method originalMethod = class_getInstanceMethod(self, @selector(viewWillAppear:));
    
    // 获取要交换的自定义实现方法
    Method swizzledMethod = class_getInstanceMethod(self, @selector(xxx_viewWillAppear:));
    
    // 交换方法
    BOOL didAddMethod = class_addMethod(self, @selector(viewWillAppear:), method_getImplementation(swizzledMethod), method_getTypeEncoding(swizzledMethod));
    
    if (didAddMethod) {
        // 如果方法添加成功,说明原来类没有实现 viewWillAppear,直接使用交换后的方法
        class_replaceMethod(self, @selector(xxx_viewWillAppear:), method_getImplementation(originalMethod), method_getTypeEncoding(originalMethod));
    } else {
        // 如果原有的方法已经实现,则交换两个方法的实现
        method_exchangeImplementations(originalMethod, swizzledMethod);
    }
}

- (void)xxx_viewWillAppear:(BOOL)animated {
    // 在这里可以调用自己的实现
    [self xxx_viewWillAppear:animated]; // 注意:这里调用的是交换后的实现

    // 自定义代码,或者添加日志、调试等
    NSLog(@"viewWillAppear 被交换了!");
}

@end

关键点

  • class_getInstanceMethod:用来获取类的实例方法。
  • method_exchangeImplementations:用来交换两个方法的实现。
  • class_addMethod:如果目标方法不存在,可以添加一个新的方法。
  • method_getImplementationmethod_getTypeEncoding:分别获取方法的实现和类型编码。

注意事项

  • 递归调用:在 xxx_viewWillAppear: 方法内部,我们使用了 self xxx_viewWillAppear:animated; 来避免递归调用自身。因为方法已经交换,这行代码会触发原始的 viewWillAppear: 方法调用。
  • 调用顺序:交换方法的顺序是非常重要的。在我们的例子中,我们在自定义方法中先执行了交换后的实现 (xxx_viewWillAppear:),然后再执行原始的 viewWillAppear:。这样可以确保自定义的行为不会丢失。

实际应用

此类应用比较多的就是数据统计,比如统计页面浏览、元素点击,通过方法交换植入监控程序,大大提高开发效率。

相关推荐
米西米西2 小时前
iOS/Swift 头像轮播组件
ios
无知的前端3 小时前
一文读懂 iOS 程序生命周期和视图生命周期
ios·面试·swift
CocoaKier3 小时前
Xcode16踩坑:UIApplication.openURL(_:)方法已彻底废弃
ios·xcode·apple
Mr.NickJJ4 小时前
iOS底层原理系列02-深入了解Objective-C
ios·objective-c·cocoa
山水域6 小时前
串行队列(Serial Queue)和并行队列(Concurrent Queue)详解
ios·swift
Swift社区8 小时前
Swift 并发中的任务让步(Yielding)和防抖(Debouncing)
开发语言·ios·swift
#摩斯先生13 小时前
IOS Xcode Could not find a storyboard named ‘Main‘ in bundle NSBundle
macos·ios·xcode
二流小码农1 天前
鸿蒙开发:ArkTs语言注释
android·ios·harmonyos
二流小码农1 天前
鸿蒙开发:权限授权封装
android·ios·harmonyos
月白星兮1 天前
IOS兼容 - uniapp ios固定定位失效与刘海屏的坑
ios·uni-app·cocoa