AOP切面编程在android上的应用

前言

切面编程一直是一个热点的话题,这篇文章讲讲一个第三方aop库在android上的应用。

本篇文章参考以下链接:
AOP第三方库

这章涉及到以下内容:

  1. aop库的配置
  2. aop导入问题解决
  3. aop的使用
  4. aop混淆
一. aop库的配置

在你的porject的build.gradle中做以下配置:

buildscript {
    
    repositories {
        google()
        jcenter()
    }
    dependencies {
        classpath 'com.android.tools.build:gradle:3.0.0'

        //AOP切面编程依赖
        classpath 'com.hujiang.aspectjx:gradle-android-plugin-aspectjx:1.1.1'
        classpath 'org.aspectj:aspectjtools:1.8.13'
    }
}

allprojects {
    repositories {
        google()
        jcenter()
    }
}

task clean(type: Delete) {
    delete rootProject.buildDir
}

在 app module 的buildle.gradle中做如下配置:

 //AOP切面编程依赖
    compile 'com.safframework:saf-aop:1.2.0'
    //rxjava2.x
    compile 'io.reactivex.rxjava2:rxandroid:2.0.1'
    compile 'io.reactivex.rxjava2:rxjava:2.1.0'
    //AOP切面编程Cache依赖
    implementation 'com.safframework:saf-cache:1.1.0'
    implementation 'tony-common:tony-common-utils:1.2.4'
    //AOP切面编程Log依赖
    implementation 'com.safframework.log:saf-log:1.5.0'
    //AOP切面编程AppPrefs依赖
    implementation 'com.safframework.injectview:saf-injectview:1.0.0'
    annotationProcessor 'com.safframework.injectview:saf-injectview-compiler:1.0.0'

需要注意的是,AOP库使用的时候,也要添加RxJava2.X的引用。

二. aop导入问题解决

在导入切面编程库依赖的时候,可能会出现如下问题:

Error:Execution failed for task ':app:transformClassesWithExtractJarsForDebug'.
> Unexpected scopes found in folder 'C:\my_demo_test\TestDemo\app\build\intermediates\transforms\AspectTransform\debug'. 
Required: SUB_PROJECTS. Found: 
EXTERNAL_LIBRARIES, PROJECT, SUB_PROJECTS

解决方案如下:

造成上述问题是由于as版本,需将Android studio的instant run关闭。具体如下: 
Settings → Build, Execution, Deployment → Instant Run and uncheck Enable Instant Run.

配图如下:

AOP切面编程在android上的应用_第1张图片
image.png

解决此方法,大家也可以参考链接
android aspectJ报错“transformClassesWithExtractJarsForDebug“

三. aop的使用

AOP涉及的注解有以下几个:


AOP切面编程在android上的应用_第2张图片
image.png

下面对各注解依次讲解:

3.1 Async异步注解

使用类似如下:

  @Async
    private void AsyncTest(){
        LogUtil.e("====当前线程===="+Thread.currentThread().getId());
        LogUtil.e("====主线程===="+ Looper.getMainLooper().getThread().getId());
    }

在主线程中调用 AsyncTest 方法,会发现其运行在一个不同于主线程的子线程中。

3.2 Cacheable缓存注解

存储数据如下:

    @Cacheable(key = "name")
    private String putCache(){
        return "我是谁";
    }

获取缓存数据:

  private String getCache(){
        Cache cache=Cache.get(CacheActivity.this);
        //Cache的存取方式就是Acache,只不过存储地址改成了缓存而已
        String name=cache.getString("name");
        return name;
    }
3.3 Log注解

你可以类似这样使用:

    @LogMethod
    private void log1(){
        LogUtil.e("=====log1======");
    }

    @LogMethod
    private String log2(String s){
        LogUtil.e("=====log2======"+s);
        return s;
    }

然后在需要用到的地方直接调用 log1 和 log2 方法就可以打印log了,它最大的好处是不会在代码中穿插各种 Log代码。

3.4 Hook注解

使用范例:

   @HookMethod(beforeMethod = "before",afterMethod = "after")
   private void getMachine(){
       LogUtil.e("======getMachine=====");
   }

   private void before(){
       LogUtil.e("====开始时间=======");
   }

    private void after(){
        LogUtil.e("====结束时间=======");
    }

然后在调用getMachine方法的时候,你会收到这样的log


image.png

这个可以用来监测一个方法执行的用时,当然,你在声明Hook注解的时候必须包含 beforeMethod,afterMethod中的一个或同时具有,若不声明,则Hook使用无效

3.5 Prefs注解

类似手机上的文件存储,存值你可以这样操作:

  @Prefs(key = "key")
    private String putPrefs(){
        return "大家好";
    }

取值是这样的:

  private String getPrefs(){
        AppPrefs appPrefs=AppPrefs.get(PrefsActivity.this);
        String key=appPrefs.getString("key",null);
        return key;
    }
3.6 safe注解

保证代码执行安全的,若不想代码抛出异常,你可以这样处理:

   @Safe
    private void testStringLength(){
        String msg=null;
        int length=msg.length();
    }

这样,你在调用 testStringLength 方法的时候就不会报错了。

3.6 Trace注解

可以用来监测方法执行时长,使用时类似这样:

  @Trace
    private void testTrace(){

        Observable.create(new ObservableOnSubscribe() {
            @Trace
            @Override
            public void subscribe(@NonNull ObservableEmitter e) throws Exception {

                e.onNext("111");


            }
        }).subscribe(new Consumer() {

            @Trace
            @Override
            public void accept(@NonNull String str) throws Exception {

            }
        });
    }

调用 testTrace 方法之后,在log 的 i 级别中,你可以看到这样的log


AOP切面编程在android上的应用_第3张图片
image.png

这里,我们就可以很清楚的看到每个方法的执行时间了。

四. 混淆

混淆的话,可以这样处理

-keep class com.safframework.aop.** { *; }

ok,今天就讲到这里,谢谢诶。

你可能感兴趣的:(AOP切面编程在android上的应用)