runtime hook 之NSDictionary、NSArray

本文探讨在处理JSON数据时遇到的nil、null、空字符串问题,通过Runtime Hook技术,避免在使用objectForKey时出现意外错误,简化类型检查的过程。

现在遇到json数据解析,服务器有可能给nil、null、@”“等,用objectForKey取出来是可以为nil等上述类型,不会报错,但是用的时候GG思密达了,而且以前自己没有类型判断完整,这时候不想去挨个挨个的改,hook大法再次登场

@implementation NSDictionary (runtimeObjectForKey)
+ (void) load{
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        Class class = [self class];
        // 选择器
        SEL originalSEL = @selector(objectForKey:);//@selector(objectForKey:);
        SEL SwizzledSEL = @selector(hObjectForKey:);


        //方法    使用NSDictionary、NSArray,必须使用与其对应的加入内存时的元类__NSDictionaryI  __NSArrayI
        //       __NSDictionaryI——NSDictionary   __NSDictionaryM——NSMutableDictionary
        //       __NSArrayI——NSArray             __NSArrayM——NSMutableArray
        Method originalMethod = class_getInstanceMethod(NSClassFromString(@"__NSDictionaryI"), originalSEL);//class_getClassMethod(class, originalSEL);备注的是获取静态方法
        Method SwizzledMethod = class_getInstanceMethod(NSClassFromString(@"__NSDictionaryI"), SwizzledSEL);//class_getClassMethod(class, SwizzledSEL);

        // 方法的实现
        IMP originalIMP = method_getImplementation(originalMethod);//class_getMethodImplementation(class, originalSEL);
        IMP SwizzledIMP = method_getImplementation(SwizzledMethod);//class_getMethodImplementation(class, SwizzledSEL);


        // 是否添加成功方法:添加了初始方法,实现内容指向目标方法体
        BOOL isSuccess = class_addMethod(class, originalSEL, SwizzledIMP, method_getTypeEncoding(SwizzledMethod));

        if (isSuccess) {
            // 初始指向目标,那么把目标的内容指向初始
            class_replaceMethod(class, SwizzledSEL, originalIMP, method_getTypeEncoding(originalMethod));
        }
        else{
            // 没有添加成功说明已经存在,就交换
            // 注意,这里交换的是IMP 实现
            method_exchangeImplementations(originalMethod, SwizzledMethod);
        }

    });
}
- (id)hObjectForKey:(id)aKey{

}
@end
### Objective-C Runtime Hook 方法的使用与实现 Objective-C 的运行时(Runtime)是一个动态的系统,允许在运行时修改类和方法的行为。通过使用 Runtime 提供的 API,可以实现方法替换(Method Swizzling)、拦截方法调用等功能。以下是关于 Objective-C Runtime Hook 方法的使用与实现的详细说明。 #### 1. 方法替换(Method Swizzling) 方法替换是通过交换两个方法的实现来改变类行为的一种技术。具体来说,可以通过 `method_exchangeImplementations` 函数实现这一功能[^3]。 ```objc #import <objc/runtime.h> @implementation MyClass (Swizzle) + (void)load { static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ Class class = [self class]; SEL originalSelector = @selector(originalMethod); SEL swizzledSelector = @selector(swizzledMethod); Method originalMethod = class_getInstanceMethod(class, originalSelector); Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector); BOOL didAddMethod = class_addMethod(class, originalSelector, method_getImplementation(swizzledMethod), method_getTypeEncoding(swizzledMethod)); if (didAddMethod) { class_replaceMethod(class, swizzledSelector, method_getImplementation(originalMethod), method_getTypeEncoding(originalMethod)); } else { method_exchangeImplementations(originalMethod, swizzledMethod); } }); } - (void)originalMethod { NSLog(@"Original Method"); } - (void)swizzledMethod { NSLog(@"Swizzled Method"); } @end ``` 上述代码展示了如何通过 `method_exchangeImplementations` 函数交换两个方法的实现。需要注意的是,为了确保安全性和正确性,通常建议在 `+load` 方法中进行方法替换[^5]。 #### 2. 获取方法实现(IMP) 通过 `methodForSelector:` 方法,可以获取某个选择器对应的实现函数指针(IMP)。以下是一个示例: ```objc #import <objc/runtime.h> id target = ...; // 假设目标对象 SEL selector = @selector(setFilled:); IMP imp = [target methodForSelector:selector]; void (*setter)(id, SEL, BOOL) = (void (*)(id, SEL, BOOL))imp; for (int i = 0; i < 1000; i++) { setter(targetList[i], selector, YES); } ``` 上述代码通过 `methodForSelector:` 获取了 `setFilled:` 方法的 IMP,并将其转换为函数指针类型以直接调用[^1]。 #### 3. KVO 的 Hook 实现 Key-Value Observing(KVO)是一种常见的观察者模式实现。通过 Runtime,可以 Hook KVO 的相关方法以实现自定义逻辑。例如,在 `didChangeValueForKey:` 方法中调用 `observeValueForKeyPath:ofObject:change:context:` 可以拦截属性变化的通知[^2]。 ```objc - (void)didChangeValueForKey:(NSString *)key { [super didChangeValueForKey:key]; // 自定义逻辑 NSLog(@"Property %@ changed", key); [self observeValueForKeyPath:key ofObject:self change:nil context:NULL]; } ``` #### 4. 注意事项 - **调试困难**:由于方法替换可能导致调用栈混乱,因此在调试时需要特别注意。建议为代码添加详细的注释和文档。 - **线程安全性**:方法替换应在 `+load` 方法中执行,以确保在类加载时完成替换,避免多线程问题。 - **递归风险**:如果新方法的实现逻辑不清晰,可能会导致递归调用。使用封装良好的方法替换工具可以有效避免这一问题[^5]。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值