RxCocoa中的AOP实现

原文出处

在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。

介绍

在RxCocoa的代码中,对于NSObject的扩展中,有这两个代码,实现了对于实例方法的AOP

public func sentMessage(_ selector: Selector) -> Observable<[Any]>
public func methodInvoked(_ selector: Selector) -> Observable<[Any]>

在本质上说,他们对selector的触发时机进行监听,并分别在selector触发前、触发后插入自定义的方法。

下面是具体的使用场景:

import UIKit
import RxSwift
import RxCocoa

class ViewController: UIViewController {
    private let bag = DisposeBag()

    override func viewDidLoad() {
        self.rx.sentMessage(#selector(ViewController.viewWillAppear(_:))).asObservable().subscribe { (_) in
            print("sendMessage")
        }.disposed(by: bag)
        
        self.rx.methodInvoked(#selector(ViewController.viewWillAppear(_:))).asObservable().subscribe { (_) in
            print("methodInvoked")
        }.disposed(by: bag)
    }
    
    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)
        print("viewWillAppear")
    }
}

输出结果


输出结果

最终的效果是在该方法的执行前与执行后的两个时间点插入了其他的方法。

下面我们就剖析一下,RxCocoa究竟做了些什么。
本文会从外部调用的方法setMessage开始,一步步查看内部的具体的实现。

Let's go

简单剖析一下上面的代码,它们分别是对于两个Observable进行了监听(RxSwift的介绍)。换句话说,在实际的实现过程中,
selector对应的方法在即将执行前肯定是要触发setMessage所对应的Observable并发出信号。
selector对应的方法在执行完成肯定要触发methodInvoked所对应的Observable并发出信号。

当然,这都是我们基于当前的调用方法给出的推测,那看看实际都做了些什么。

首先看setMessage所对应的方法的具体实现。

   public func sentMessage(_ selector: Selector) -> Observable<[Any]> {
        return synchronized {
            // in case of dealloc selector replay subject behavior needs to be used
            if selector == deallocSelector {
                return deallocating.map { _ in [] }
            }

            do {
                let proxy: MessageSentProxy = try registerMessageInterceptor(selector) 
                return proxy.messageSent.asObservable()
            }
            catch let e {
                return Observable.error(e)
            }
        }
    }

上面的代码有2行是核心。

let proxy: MessageSentProxy = try registerMessageInterceptor(selector)
proxy.messageSent.asObservable()

其中第一行代码实现的功能是: 为该selector创建或者获取一个代理对象
第二行代码,则是返回messageSent的信号源,它能保证在恰当的时候出发信号。

下面分别要对这两个代码进行具体的分析

MessageSentProxy

贴一下MessageSentProxy的具体的代码

fileprivate final class MessageSentProxy
        : MessageInterceptorSubject
        , RXMessageSentObserver {
        typealias E = [AnyObject]

        let messageSent = PublishSubject<[Any]>() //message所对应的subject(Observable),也就是被观察者
        let methodInvoked = PublishSubject<[Any]>()//invoked 所对应的被观察者

        @objc var targetImplementation: IMP = RX_default_target_implementation()

        var isActive: Bool {
            return targetImplementation != RX_default_target_implementation()
        }

        init() {
        }

        @objc func messageSent(withArguments arguments: [Any]) -> Void {
            messageSent.on(.next(arguments))
        }

        @objc func methodInvoked(withArguments arguments: [Any]) -> Void {
            methodInvoked.on(.next(arguments))
        }

        deinit {
            messageSent.on(.completed)
            methodInvoked.on(.completed)
        }
    }

从上面可以看出,MessageSentProxy这个类分别持有了名为messageSent以及methodInvoked的两个被观察者。
应该可以看出,他们分别对应着selector前后的触发时机,对这两个参数进行监听,也就是变相的监听了selector触发的前后的时机。

其中内部的messageSent(withArguments arguments: [Any])以及methodInvoked(withArguments arguments: [Any])方法分别促使两个被观察者发出信号。

两个方法分别用@objc进行修饰,标明两个方法都可以被OC的调用机制触发。

那当前可以整理出的触发顺序是:

  1. MessageSentProxy中messageSent(withArguments arguments: [Any])发放被调用
  2. MessageSentProxy中messageSent = PublishSubject<[Any]>()触发onNext信号
  3. 外部对于messageSent的监听事件触发

那好,继续深究一下。

registerMessageInterceptor(selector)

上面讲到registerMessageInterceptor(selector)方法是通过selector来创建一个MessageSentProxy,用来实现Observable。 由于同一个selector可能有2个监听事件sentMessage以及methodInvoke, 而两者都是通过同一个MessageSentProxy来实现。因此最终一个selector对应于一个MessageSentProxy,也不需要进行反复的创建。
那究竟做了哪些事情呢?

 //该方法保证了,在同一个target上 同一个selector 只会有 一个MessageInterceptorSubject与之绑定。
    fileprivate func registerMessageInterceptor<T: MessageInterceptorSubject>(_ selector: Selector) throws -> T {
        let rxSelector = RX_selector(selector) //添加前缀,生成一个新的selector 当前只有名字
        let selectorReference = RX_reference_from_selector(rxSelector) 

        let subject: T
        if let existingSubject = objc_getAssociatedObject(base, selectorReference) as? T { //如果这个selector有关联值的话, 赋值给subject
            subject = existingSubject
        }
        else { //如果没有,则创建一个Subject对象,把它关联给当前的base
            subject = T()
            objc_setAssociatedObject(
                base,
                selectorReference,
                subject,
                .OBJC_ASSOCIATION_RETAIN_NONATOMIC
            )
        }

        if subject.isActive { // 表示其内部的IMP 其实已经赋值了, 否则就是还没有赋值。
            return subject
        }

        var error: NSError?
        let targetImplementation = RX_ensure_observing(base, selector, &error)
        if targetImplementation == nil {
            throw error?.rxCocoaErrorForTarget(base) ?? RxCocoaError.unknown
        }

        subject.targetImplementation = targetImplementation!

        return subject

上面的代码做了大概几件事:
1.将MessageInterceptorSubject,也就是该方法调用处的MessageSentProxy通过selector与当前的对象进行绑定。

2.如果对应的selector已经绑定过了,则获取已绑定的MessageSentProxy。
这是由于对应同一个selector,我们可能有sentMessage以及methodInvoke两个监听事件,两个监听事件由于可以使用同一个MessageSentProxy,避免重复工作。
3.开始hook对应的selector,进行插入操作
核心代码
let targetImplementation = RX_ensure_observing(base, selector, &error)

解读方法的hook

这也是当前这个过程中最核心的部分。
还记得KVO是怎么实现的吗?如果不记得,可以在看一下(KVO代码测试以及探究)

继续向下深究RX_ensure_observing(base, selector, &error),并且删除用于行,留下核心代码

/**
 This is the main entry point for observing messages sent to arbitrary objects.
 */
-(IMP __nullable)ensurePrepared:(id __nonnull)target forObserving:(SEL __nonnull)selector error:(NSErrorParam)error {
    Method instanceMethod = class_getInstanceMethod([target class], selector); //根据target sel 获取method

        Class __nullable swizzlingImplementorClass = [self prepareTargetClassForObserving:target error:error];
        if (swizzlingImplementorClass == nil) {
            return nil;
        }

        NSString *methodEncoding = RX_method_encoding(instanceMethod);
        RXInterceptWithOptimizedObserver optimizedIntercept = optimizedObserversByMethodEncoding[methodEncoding];

        if (!RX_method_has_supported_return_type(instanceMethod)) {
            RX_THROW_ERROR([NSError errorWithDomain:RXObjCRuntimeErrorDomain
                                               code:RXObjCRuntimeErrorObservingMessagesWithUnsupportedReturnType
                                           userInfo:nil], nil);
        }

        // optimized interception method
        if (optimizedIntercept != nil) {
            IMP interceptorIMPForSelector = [self interceptorImplementationForSelector:selector forClass:swizzlingImplementorClass];
            if (interceptorIMPForSelector != nil) {
                return interceptorIMPForSelector;
            }

            if (!optimizedIntercept(self, swizzlingImplementorClass, selector, error)) {
                return nil;
            }

            interceptorIMPForSelector = [self interceptorImplementationForSelector:selector forClass:swizzlingImplementorClass];
            if (interceptorIMPForSelector != nil) {
                return interceptorIMPForSelector;
            }
        }
        // default fallback to observing by forwarding messages
        else {
            if ([self forwardingSelector:selector forClass:swizzlingImplementorClass]) {
                return RX_default_target_implementation();
            }

            if (![self observeByForwardingMessages:swizzlingImplementorClass
                                          selector:selector
                                            target:target
                                             error:error]) {
                return nil;
            }

            if ([self forwardingSelector:selector forClass:swizzlingImplementorClass]) {
                return RX_default_target_implementation();
            }
        }
    

    RX_THROW_ERROR([NSError errorWithDomain:RXObjCRuntimeErrorDomain
                                       code:RXObjCRuntimeErrorUnknown
                                   userInfo:nil], nil);
}

那实际是如何fake出一个新的派生的class来取代原来的class呢?

-(Class __nullable)prepareTargetClassForObserving:(id __nonnull)target error:(NSErrorParam)error

这个方法做了一下几件事
1.如果已经有了, 则直接返回(通过关联值进行关联)
2.如果class方法与object_getClass获取的元类型不一样,判断是否为KVO,若不是,默认无法处理,返回错误
3.生成最终的fakeClass, 并与关联类型进行绑定。

其中,这个fakeClass就类似于KVO机制中的NSKVONotifying_类名. 都是起到一个替换的作用,让实际方法执行的selector被重写,执行的起始以及结尾分别加入触发。

那说到底,还是没有说,具体这两个方法在细节上是如何插入对应的节点的(捂脸)。

有关于fakeClass的具体实现过程,我会在下一篇文章中进行详细的描述。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,313评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,369评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,916评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,333评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,425评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,481评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,491评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,268评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,719评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,004评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,179评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,832评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,510评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,153评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,402评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,045评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,071评论 2 352

推荐阅读更多精彩内容

  • 在正文开始之前的最后,放上GitHub链接和引入依赖的gradle代码: Github: https://gith...
    苏苏说zz阅读 677评论 0 2
  • 设计模式是什么? 你知道哪些设计模式,并简要叙述? 设计模式是一种编码经验,就是用比较成熟的逻辑去处理某一种类型的...
    iOS菜鸟大大阅读 705评论 0 1
  • 1.设计模式是什么? 你知道哪些设计模式,并简要叙述? 设计模式是一种编码经验,就是用比较成熟的逻辑去处理某一种类...
    司马DE晴空阅读 1,285评论 0 7
  • 响应式编程简介 响应式编程是一种基于异步数据流概念的编程模式。数据流就像一条河:它可以被观测,被过滤,被操作,或者...
    说码解字阅读 3,060评论 0 5
  • 此生自在逍遥 我大概再无法逍遥了吧 自在 现在看起来似乎也是很难得的东西 那些逍遥自在的人必定都一颗自爱自强的自尊...
    匕禾页1998阅读 279评论 0 2