美文网首页程序员
快速掌握Android AOP(一)

快速掌握Android AOP(一)

作者: sys_out | 来源:发表于2019-01-16 16:38 被阅读14次

    详细讲解请查看:https://blog.csdn.net/innost/article/details/49387395

    什么是AOP编程?

    简单的说:Aop编程就是在各个模块像"插桩"一样“切”入我们的代码逻辑,即面向切面编程。
    例子:比如我们要在每个方法中都打印该方法运行信息的log,传统方式会在每个方法中都写打印的逻辑。显然,这种方式不可取。Aop就是向各个方法的插入打印的逻辑。

    如何AOP编程?

    Aop编程可以通过很多方法 有apt(自动生成源码) 、aspectJ(编译期插入字节码)、asm/javassit(操作字节码,前者效率高但是是字节码层面的操作,后者效率低一点但是是封装好的api)、xposed、dexPosed、动态代理...来实现。aop是一种编程思想,所以方式不限。这就类似oop编程,语言不是重点。

    通过这篇文章可了解前三种方式:https://www.jianshu.com/p/dca3e2c8608a?from=timeline

    我们重点关注AspectJ这种android流行的实现方式。

    什么是AspectJ?

    AspectJ是一种和java完全兼容的语言。aspectJ的使用有两种方式:

    1. 完全使用AspectJ语言编写,和java语法没有区别,可以调用java代码,只是多了关键字。
    2. 使用java编写,通过@AspectJ注解java代码。
      无论使用哪种方式,最后都要通过ajc来编译,ajc也可以用来编译java源码。
    如何使用AspectJ?

    AspectJ中的一些概念:

    • Join Points
      指的是程序中的执行点。只有确认执行点,才能够确定切面位置。但并不是所有点都可以作为执行点。以下下点可以作为执行点:


      图片来自https://blog.csdn.net/innost/article/details/49387395
    • PointCuts
      如果把JoinPoints比做成程序调试时的断点,那么PointCuts就相当于给断点设置的条件。例如这样:
    public pointcut  testAll(): call(public  *  *.println(..)) && !within(TestAspect) ;
    

    这段代码的含义就是:名字叫testAll的PointCut,不限包名、类名、返回值类型、参数名、方法名为println()。并且不是TestAspect类型的。
    pointCuts通配符类型如下:

    1. PointCuts对应join Points类型


      图片来自https://blog.csdn.net/innost/article/details/49387395

      在指定好相应的join Points类型后,还有MethodSignature,ConstructorSignature,TypeSinature,FieldSignatur等需要明确

    2. MethodSignature:“*”表示任意字符,“..”表示子类包名,"+"表示子类,".."表示任意类型参数。具体如下(如果想快速掌握,下面可粗略看看)
    @注解 访问权限 返回值的类型 包名.函数名(参数)
      @注解和访问权限(public/private/protect,以及static/final)属于可选项。如果不设置它们,则默认都会选择。以访问权限为例,如果没有设置访问权限作为条件,那么public,private,protect及static、final的函数都会进行搜索。
      返回值类型就是普通的函数的返回值类型。如果不限定类型的话,就用*通配符表示
      包名.函数名用于查找匹配的函数。可以使用通配符,包括*和..以及+号。其中*号用于匹配除.号之外的任意字符,而..则表示任意子package,+号表示子类。
         比如:
         java.*.Date:可以表示java.sql.Date,也可以表示java.util.Date
         Test*:可以表示TestBase,也可以表示TestDervied
         java..*:表示java任意子类
         java..*Model+:表示Java任意package中名字以Model结尾的子类,比如TabelModel,TreeModel
         等
      最后来看函数的参数。参数匹配比较简单,主要是参数类型,比如:
         (int, char):表示参数只有两个,并且第一个参数类型是int,第二个参数类型是char
         (String, ..):表示至少有一个参数。并且第一个参数类型是String,后面参数类型不限。在参数匹配中,
         ..代表任意参数个数和类型
         (Object ...):表示不定个数的参数,且类型都是Object,这里的...不是通配符,而是Java中代表不定参数的意思
    
    1. ConstructorSignature:和MethodSignature类似,只是没有返回值,并且方法名为new。具体如下:
    public *..TestDerived.new(..):
      public:选择public访问权限
      *..代表任意包名
      TestDerived.new:代表TestDerived的构造函数
      (..):代表参数个数和类型都是任意
    再来看Field Signature和Type Signature,用它们的地方见图5。下面直接上几个例子:
    Field Signature标准格式:
    @注解 访问权限 类型 类名.成员变量名
      其中,@注解和访问权限是可选的
      类型:成员变量类型,*代表任意类型
      类名.成员变量名:成员变量名可以是*,代表任意成员变量
    比如,
    set(inttest..TestBase.base):表示设置TestBase.base变量时的JPoint
    Type Signature:直接上例子
    staticinitialization(test..TestBase):表示TestBase类的static block
    handler(NullPointerException):表示catch到NullPointerException的JPoint。注意,图2的源码第23行截获的其实是Exception,其真实类型是NullPointerException。但是由于JPointer的查询匹配是静态的,即编译过程中进行的匹配,所以handler(NullPointerException)在运行时并不能真正被截获。只有改成handler(Exception),或者把源码第23行改成NullPointerException才行。
    
    1. 其他匹配方法


      图片来自https://blog.csdn.net/innost/article/details/49387395
    • advice和aspect
      上面我们知道了如何选择joinPoint,那么如何插入我们的逻辑代码呢?
      advice的类型:


      image.png

      around是替换方法,理论上around是可以代替before和after的。在around中可以通过proceed()调用实际方法。
      代码示例:


      image.png

    aspectJ文件的定义:

    public aspect 名字 {//aspect关键字和class的功能一样,文件名以.aj结尾
     pointcuts定义...
     advice定义...
    }
    

    我们定义一个LogAspect,在LogAspect中,我们在关键JPoint上设置advice,这些advice就是打印日志
    Aop的精髓就是:我们不需要在相应方法中打印log或检查权限,只要将逻辑写到对应的AspectJ文件中。

    注意,读者在把玩代码时候,一定会碰到AspectJ语法不熟悉的问题。所以请读者记得随时参考官网的文档。这里有一个官方的语法大全:
    http://www.eclipse.org/aspectj/doc/released/quick5.pdf 或者官方的另外一个文档也可以:
    http://www.eclipse.org/aspectj/doc/released/progguide/semantics.html

    目前先写到这里,后面继续更新。其实这篇就是最上面那篇文章的简化版,当然如果有我自己的理解,我也会加上去。

    相关文章

      网友评论

        本文标题:快速掌握Android AOP(一)

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