Rachook伎俩

2018-05-29  本文已影响23人  我是小胡胡分胡

signalForSelector

跟系统的kvo实现原理类似。
3种方式

1, rac_signalForSelector:

原理:
1、创建子类, 对应为isa的名字,加上后缀_RACSelectorSignal
( 类 : class返回类名, isa指向的类名, 需要注意这两种情况的区别。
当一个类注册kvo观察了, class返回的是原有类名, isa执行了新的之类。这种情况就是两种不同的返回。 )

2、重写子类forwardInvocation方法。
当调用子类的forwardInvocation选择子时,调用新的newForwardInvocation block。 在新的newForwardInvocation block中调用RACForwardInvocation函数,在全局RACForwardInvocation 中,invoke原有选择子对应的消息。 然后 给订阅者发送消息。
如果没有实现选择子, doesNotRecognizeSelector。
如果子类没有实现forwardInvocation,给父类发送 forwardInvocation, 如果实现了forwardInvocation ,则给这个子类发送forwardInvocation

3、重写子类respondsToSelector方法。

4、重写子类(通过class获取的类名字)的class方法, 让其返回原有的类名字。

5、重写子类(通过isa获取的类名字)的class方法,让其返回原有的类名字。

6、重写子类的methodSignatureForSelector方法。使其给给原有类发送methodSignatureForSelector消息。

7、将当前recevier的类型设置为子类

8、重写传进来的selector的方法,让其去调用forwardInvocation。

selector在receiver的类中实现了:
aliasSelector 绑定到原有方法。
重写selector,让其走forwardInvocation消息转发。

先走原有方法--》 再发送信号。

selector在receiver的类中没有实现:
//如果实现了, 就能转化参数。
//如果没有实现, 不能转化,会产生异常崩溃。因为RACSignatureForUndefinedSelector统一当作id类型的

@protocol MyAddProtocol<NSObject>
@optional
-(void)aa1:(NSString *) hello color:(UIColor *)color int:(NSInteger )intVlaue;
@end

#import "hookProxyDelegateViewController.h"

@interface hookProxyDelegateViewController () <UITableViewDelegate,UITableViewDataSource, MyAddProtocol>

-(void) test1 {
    [[self rac_signalForSelector:@selector(aa1:color:int:)]subscribeNext:^(RACTuple * _Nullable x) {
        NSLog(@"%@", x);
    } ];
    [self aa1:@"asdfa" color:[UIColor redColor] int:122];
}
//如果实现了, 就能转化参数。
//如果没有实现, 不能转化,会产生异常崩溃。因为RACSignatureForUndefinedSelector统一当作id类型的
-(void)aa1:(NSString *)hello color:(UIColor *)color int:(NSInteger)intVlaue{
    NSLog(@"asdfa");
}

2, rac_signalForSelector: fromProtocol:

selector在receiver的类中实现了:
跟rac_signalForSelector:方法效果相同。

selector在receiver的类中没有实现:

检查在协议中selector方法是否声明了, 现在必须协议方法列表中找,然后在不是可选协议方法列表中找。 都没有找到。 报异常抛出。selector和协议不匹配。
重写selector,让其走forwardInvocation消息转发。

3, [[RACDelegateProxy alloc] initWithProtocol:]

racdelegateProxy 模式改变身份
swizzle racdelegateProxy类的isa

swizzleDeallocIfNeed

不会swizzle isa

//看到这里,是用__block声明的 originalDealloc是一个函数指针
__block void (*originalDealloc)(__unsafe_unretained id, SEL) = NULL;
id newDealloc = ^(__unsafe_unretained id self) {

//hook住的目的就是为了要摄入dispose
[compoundDisposable dispose];

block中捕获了originalDealloc, originalDealloc指针本身的地址拷贝到堆上去了
执行dealloc
msgSend(&superInfo, deallocSelector);
或者originalDealloc(self, deallocSelector);

}

可以看到 &originalDealloc的地址因为block的捕获,地址变化了

if (!class_addMethod(classToSwizzle, deallocSelector, newDeallocIMP, "v@:"))

//the class already contains a method implementation with that name
//本类中实现了名字为dealloc的选择子

调dealloc方法,会触发到newDeallocIMP的方法指针调用
originalDealloc = (typeof(originalDealloc))method_setImplementation(deallocMethod, newDeallocIMP);

}
else {
// 增加成功了。本类没有实现dealloc方法
当执行dealloc时,就会执行newDealloc

}

//hook操作 Sets the implementation of a method.The previous implementation of the method.
//返回原来的函数指针, 保存原来的dealloc函数指针IMP到originalDealloc, 便于block里面调用
//调用Method时候,会调用newDeallocIMP指针,也就是新的deallocblock
originalDealloc = (typeof(originalDealloc))method_setImplementation(deallocMethod, newDeallocIMP);

所以,跟我们常见的swizzle不同的是,
方法交换,调A就是调B(B里面在调B,此时方法交换了,所以就能回调到原来的A),调B调就是A;
而上面的swillzeDealloc 不是交换。 只是一个调剂手段。 A调B(B里面调用A)

上一篇下一篇

猜你喜欢

热点阅读