Message forwarding mechanism (消息转发机制)

iOS的消息转发机制

iOS的消息转发机制是在消息发送给对象时,找不到对应的实例方法的情况下启动的。消息转发允许对象在运行时处理无法识别的消息,提供了一种动态的、灵活的消息处理方式。

消息转发机制主要分为三个阶段:

  1. 动态方法解析
  2. 快速转发
  3. 标准转发

1. 动态方法解析

在这个阶段,运行时系统会询问对象是否能动态添加方法来处理未知消息。可以通过实现 +resolveInstanceMethod:+resolveClassMethod: 方法来动态添加方法实现。

objc 复制代码
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    if (sel == @selector(someMethod)) {
        class_addMethod([self class], sel, (IMP)someMethodImplementation, "v@:");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}

void someMethodImplementation(id self, SEL _cmd) {
    NSLog(@"Dynamically added method");
}

2. 快速转发

如果动态方法解析失败,运行时会调用 -forwardingTargetForSelector: 方法,询问对象是否有其他对象可以处理该消息。通过返回另一个对象的实例,消息可以转发给该实例。

objc 复制代码
- (id)forwardingTargetForSelector:(SEL)aSelector {
    if (aSelector == @selector(someMethod)) {
        return alternateObject;
    }
    return [super forwardingTargetForSelector:aSelector];
}

3. 标准转发

如果前两个阶段都失败,运行时会调用 -methodSignatureForSelector:-forwardInvocation: 方法进行标准消息转发。首先,通过 -methodSignatureForSelector: 方法获取方法签名,然后在 -forwardInvocation: 方法中处理消息。

objc 复制代码
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    if (aSelector == @selector(someMethod)) {
        return [NSMethodSignature signatureWithObjCTypes:"v@:"];
    }
    return [super methodSignatureForSelector:aSelector];
}

- (void)forwardInvocation:(NSInvocation *)anInvocation {
    if ([alternateObject respondsToSelector:[anInvocation selector]]) {
        [anInvocation invokeWithTarget:alternateObject];
    } else {
        [super forwardInvocation:anInvocation];
    }
}

图示

为了更好地理解这个过程,下面是一个示意图:

plaintext 复制代码
+----------------------+
|        Sender        |
+----------------------+
            |
            | Message: someMethod
            v
+----------------------+
|     Receiver         |
+----------------------+
|                      |
| -resolveInstanceMethod:
|     - Can add method? ----> YES (add method)
|                      |
|                      v
|  NO                  |
|                      |
| -forwardingTargetForSelector:
|     - Other object to handle? ----> YES (forward to other object)
|                      |
|                      v
|  NO                  |
|                      |
| -methodSignatureForSelector:
|     - Get method signature ----> Signature or nil
|                      |
|                      v
|  Signature           |
|                      |
| -forwardInvocation:
|     - Handle invocation ----> Forward or error
+----------------------+

示例代码

下面是一个完整的示例代码,展示如何使用消息转发机制:

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

@interface SQIAlternateObject : NSObject
- (void)someMethod;
@end

@implementation SQIAlternateObject
- (void)someMethod {
    NSLog(@"AlternateObject handling someMethod");
}
@end

@interface MyObject : NSObject
@end

@implementation MyObject {
    SQIAlternateObject *alternateObject;
}

- (instancetype)init {
    self = [super init];
    if (self) {
        alternateObject = [[SQIAlternateObject alloc] init];
    }
    return self;
}

+ (BOOL)resolveInstanceMethod:(SEL)sel {
    if (sel == @selector(someMethod)) {
        class_addMethod([self class], sel, (IMP)someMethodImplementation, "v@:");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}

void someMethodImplementation(id self, SEL _cmd) {
    NSLog(@"Dynamically added method");
}

- (id)forwardingTargetForSelector:(SEL)aSelector {
    if (aSelector == @selector(someMethod)) {
        return alternateObject;
    }
    return [super forwardingTargetForSelector:aSelector];
}

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    if (aSelector == @selector(someMethod)) {
        return [NSMethodSignature signatureWithObjCTypes:"v@:"];
    }
    return [super methodSignatureForSelector:aSelector];
}

- (void)forwardInvocation:(NSInvocation *)anInvocation {
    if ([alternateObject respondsToSelector:[anInvocation selector]]) {
        [anInvocation invokeWithTarget:alternateObject];
    } else {
        [super forwardInvocation:anInvocation];
    }
}

@end

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        MyObject *obj = [[MyObject alloc] init];
        [obj someMethod];
    }
    return 0;
}

在这个示例中,我们展示了如何通过动态方法解析、快速转发和标准转发来处理 someMethod 方法。

相关推荐
文件夹__iOS4 小时前
AsyncStream 进阶实战:SwiftUI 全局消息流极简实现
ios·swiftui·swift
2501_916008896 小时前
深入解析iOS机审4.3原理与混淆实战方法
android·java·开发语言·ios·小程序·uni-app·iphone
忆江南6 小时前
Flutter深度全解析
ios
山水域6 小时前
Swift 6 严格并发检查:@Sendable 与 Actor 隔离的深度解析
ios
楚轩努力变强7 小时前
iOS 自动化环境配置指南 (Appium + WebDriverAgent)
javascript·学习·macos·ios·appium·自动化
游戏开发爱好者81 天前
日常开发与测试的 App 测试方法、查看设备状态、实时日志、应用数据
android·ios·小程序·https·uni-app·iphone·webview
黑码哥1 天前
ViewHolder设计模式深度剖析:iOS开发者掌握Android列表性能优化的实战指南
android·ios·性能优化·跨平台开发·viewholder
2501_915106321 天前
app 上架过程,安装包准备、证书与描述文件管理、安装测试、上传
android·ios·小程序·https·uni-app·iphone·webview
2501_915106321 天前
使用 Sniffmaster TCP 抓包和 Wireshark 网络分析
网络协议·tcp/ip·ios·小程序·uni-app·wireshark·iphone
熊猫钓鱼>_>1 天前
移动端开发技术选型报告:三足鼎立时代的开发者指南(2026年2月)
android·人工智能·ios·app·鸿蒙·cpu·移动端