aspect源码之切面编程

作者: _兜兜转转_ | 来源:发表于2019-04-24 12:00 被阅读0次

Aspect Oriented Programming(AOP),面向切面编程,是一个比较热门的话题。AOP主要实现的目的是针对业务处理过程中的切面进行提取,它所面对的是处理过程中的某个步骤或阶段,以获得逻辑过程中各部分之间低耦合性的隔离效果。比如我们最常见的就是日志记录了,举个例子,我们现在提供一个查询学生信息的服务,但是我们希望记录有谁进行了这个查询。如果按照传统的OOP的实现的话,那我们实现了一个查询学生信息的服务接口(StudentInfoService)和其实现 类 (StudentInfoServiceImpl),同时为了要进行记录的话,那我们在实现类(StudentInfoServiceImpl)中要添加其实现记录的过程。这样的话,假如我们要实现的服务有多个呢?那就要在每个实现的类都添加这些记录过程。这样做的话就会有点繁琐,而且每个实现类都与记录服务日志的行为紧耦合,违反了面向对象的规则。那么怎样才能把记录服务的行为与业务处理过程中分离出来呢?看起来好像就是查询学生的服务自己在进行,但却是背后日志记录对这些行为进行记录,并且查询学生的服务不知道存在这些记录过程,这就是我们要讨论AOP的目的所在。AOP的编程,好像就是把我们在某个方面的功能提出来与一批对象进行隔离,这样与一批对象之间降低了耦合性,可以就某个功能进行编程。

实现切面编程,拦截类的函数和实例的函数可以通过框架Aspects来实现,下面来介绍一下结构和关键实现函数。
框架的架构基本如下:

架构图

入口函数是+ (id<AspectToken>)aspect_hookSelector:(SEL)selector withOptions:(AspectOptions)options usingBlock:(id)block error:(NSError **)error;然后调用了一个c函数来实现hook的过程。
那么我们看一下hook的大概实现

static id aspect_add(id self, SEL selector, AspectOptions options, id block, NSError **error) {
//判断三个参数是否为空
    NSCParameterAssert(self);
    NSCParameterAssert(selector);
    NSCParameterAssert(block);

    __block AspectIdentifier *identifier = nil;
//加锁
    aspect_performLocked(^{
        if (aspect_isSelectorAllowedAndTrack(self, selector, options, error)) {
//获取和创建AspectsContainer,其实AspectsContainer就是存储在实例中
//的信息,包括hook的三个数组函数列表
            AspectsContainer *aspectContainer = aspect_getContainerForObject(self, selector);
            identifier = [AspectIdentifier identifierWithSelector:selector object:self options:options block:block error:error];
            if (identifier) {
//addAspect:identifier添加到aspectContainer
                [aspectContainer addAspect:identifier withOptions:options];
                // hook的实现函数
                aspect_prepareClassAndHookSelector(self, selector, error);
            }
        }
    });
    return identifier;
}

上边是准备实现hook的准备工作
利用黑魔法class_addMethodclass_replaceMethod函数实现是拦截了forwardInvocation:,根据selfaspectContainer存的函数数组来分别执行beforeinstandafter三个不同切面的函数。
拦截函数是进入到下边函数:

static void __ASPECTS_ARE_BEING_CALLED__(__unsafe_unretained NSObject *self, SEL selector, NSInvocation *invocation) {
    NSCParameterAssert(self);
    NSCParameterAssert(invocation);
    SEL originalSelector = invocation.selector;
    SEL aliasSelector = aspect_aliasForSelector(invocation.selector);
    invocation.selector = aliasSelector;
//获取objectContainer和classContainer
    AspectsContainer *objectContainer = objc_getAssociatedObject(self, aliasSelector);
    AspectsContainer *classContainer = aspect_getContainerForClass(object_getClass(self), aliasSelector);
    AspectInfo *info = [[AspectInfo alloc] initWithInstance:self invocation:invocation];
    NSArray *aspectsToRemove = nil;
    // hooks之前的函数先执行class 
    aspectsToRemove = aspect_invoke(classContainer.beforeAspects, info,aspectsToRemove);
// hooks之前的函数先执行实例
    aspectsToRemove = aspect_invoke(objectContainer.beforeAspects, info,aspectsToRemove);
    // hooks的函数执行
    BOOL respondsToAlias = YES;
    if (objectContainer.insteadAspects.count || classContainer.insteadAspects.count) {
        aspectsToRemove =aspect_invoke(classContainer.insteadAspects, info,aspectsToRemove);
        aspectsToRemove=aspect_invoke(objectContainer.insteadAspects, info,aspectsToRemove);
    }else {
        Class klass = object_getClass(invocation.target);
        do {
            if ((respondsToAlias = [klass instancesRespondToSelector:aliasSelector])) {
                [invocation invoke];
                break;
            }
        }while (!respondsToAlias && (klass = class_getSuperclass(klass)));
    }
    // hooks之后的函数执行class
    aspectsToRemove=aspect_invoke(classContainer.afterAspects, info,aspectsToRemove);
    //  hooks之后的函数执行实例
    aspectsToRemove=aspect_invoke(objectContainer.afterAspects, info,aspectsToRemove);
//如果没有钩子,则调用原始函数 一般引发异常,因为有钩子才会走到这里。
    if (!respondsToAlias) {
        invocation.selector = originalSelector;
        SEL originalForwardInvocationSEL = NSSelectorFromString(AspectsForwardInvocationSelectorName);
        if ([self respondsToSelector:originalForwardInvocationSEL]) {
            ((void( *)(id, SEL, NSInvocation *))objc_msgSend)(self, originalForwardInvocationSEL, invocation);
        }else {
            [self doesNotRecognizeSelector:invocation.selector];
        }
    }
    // 排队等待销毁执行 remove函数
    [aspectsToRemove makeObjectsPerformSelector:@selector(remove)];
}

到此已经函数执行完毕。我们可以利用Aspects实现一些东西比如:

  • 无痕埋点
  • 热更新
  • 热门数据统计

相关文章

网友评论

    本文标题:aspect源码之切面编程

    本文链接:https://www.haomeiwen.com/subject/vwwwgqtx.html