程序员

快速掌握Android AOP(一)

2019-01-16  本文已影响14人  sys_out

详细讲解请查看: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中的一些概念:

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

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

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

上一篇下一篇

猜你喜欢

热点阅读