溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Android中怎么實現(xiàn)切面編程

發(fā)布時間:2021-06-17 11:19:15 來源:億速云 閱讀:94 作者:Leah 欄目:移動開發(fā)

本篇文章給大家分享的是有關(guān)Android中怎么實現(xiàn)切面編程,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

安裝AspectJ

Android上的ApsectJ開發(fā)由幾部分組成,AspectJ gradle插件,ApsectJ依賴,還有 AspectJ編譯器。
首先安裝AspectJ編譯器很簡單,就跟安裝JAVA環(huán)境一樣,

下載鏈接:http://www.eclipse.org/downloads/download.php?file=/tools/aspectj/aspectj-1.9.0.jar

目前最新的已經(jīng)更新到1.9.1了。如果你電腦已經(jīng)有JAVA環(huán)境的話直接運行這個jar包就行,
在安裝完畢后需要配置環(huán)境變量到 aspectj的bin目錄下,這里不贅述

export PATH="$PATH:~/Library/Android/sdk/platform-tools"
export PATH="$PATH:/usr/local/opt/gradle/gradle-4.1/bin"
export PATH="$PATH:~/Library/Android/sdk/ndk-bundle"
export PATH="$PATH:~/Library/flutter/bin"
export PATH="$PATH:~/Library/kotlinc/bin"
export PATH="$PATH:~/Library/AspectJ/bin" <- AspectJ的PATH

配置完后運行 ajc -v 應該可以看到對應輸出

AspectJ Compiler 1.9.0 (1.9.0 - Built: Monday Apr 2, 2018 at 18:52:10 GMT)

配置Android Gradle增加AspectJ依賴

構(gòu)建帶AspectJ支持的Android App的流程是先按正常流程編譯出 .class 文件后,再用 ajc 編譯器在 .class文件中插入我們需要的代碼。

首先需要把 AspectJ 依賴加到 gradle根目錄中,

buildscript {
  repositories {
    google()
    jcenter()
  }
  dependencies {
    classpath 'com.android.tools.build:gradle:3.1.2'
    classpath 'org.aspectj:aspectjtools:1.8.9' //Aspect
    classpath 'org.aspectj:aspectjweaver:1.8.9' //Aspect
  }
}

然后在項目app目錄的build.gradle需要添加以下內(nèi)容,

apply plugin: 'com.android.application'
//+增加內(nèi)容
import org.aspectj.bridge.MessageHandler
import org.aspectj.tools.ajc.Main
buildscript {
  repositories {
    mavenCentral()
  }
  dependencies {
    classpath 'org.aspectj:aspectjtools:1.8.9'
    classpath 'org.aspectj:aspectjweaver:1.8.9'
  }
}
repositories {
  mavenCentral()
}

final def log = project.logger
final def variants = project.android.applicationVariants
variants.all { variant ->
  if (!variant.buildType.isDebuggable()) {
    log.debug("Skipping non-debuggable build type '${variant.buildType.name}'.")
    return;
  }

  JavaCompile javaCompile = variant.javaCompile
  javaCompile.doLast {
    String[] args = ["-showWeaveInfo",
             "-1.8",
             "-inpath", javaCompile.destinationDir.toString(),
             "-aspectpath", javaCompile.classpath.asPath,
             "-d", javaCompile.destinationDir.toString(),
             "-classpath", javaCompile.classpath.asPath,
             "-bootclasspath", project.android.bootClasspath.join(File.pathSeparator)]
    MessageHandler handler = new MessageHandler(true);
    new Main().run(args, handler);
  }
}
//-增加內(nèi)容

這段gradle腳本是在java編譯完成后追加一個 acj 的編譯流程,

MessageHandler 是 AspectJ Tools中的對象,用來接收參數(shù)然后進行 acj 編譯的。

最后再把 dependencies依賴加上對AspectJ的支持就可以了,

implementation 'org.aspectj:aspectjrt:1.9.0'

創(chuàng)建AspectJ代碼

下面這部分代碼看起來會一臉懵逼,不過目前先不用管具體的語法含義,

先跑起來環(huán)境,然后再結(jié)合理論慢慢在修改代碼中感受就能快速的上手AOP了。

以一個HelloWorld為例子,我們的MainActivity中啥事情不干,只有基本的生命周期方法,

public class MainActivity extends AppCompatActivity {
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
  }

  @Override
  protected void onStart() {
    super.onStart();
  }

  @Override
  protected void onPause() {
    super.onPause();
  }

  @Override
  protected void onStop() {
    super.onStop();
  }

  @Override
  protected void onDestroy() {
    super.onDestroy();
  }
}

現(xiàn)在我們要寫一個AspectJ類,這個類看起來會跟一般的Java類有點不同,可以理解為它只是用注解作為媒介,讓ACJ編譯器知道要去注入哪些方法。

這個類要做的事情是告訴ACJ編譯器,要在MainActivity中的每個方法前面打印一行l(wèi)og,輸出當前執(zhí)行的是哪個方法,

@Aspect
public class AspectTest {
  private static final String TAG = "AspectTest";

  @Pointcut("execution(* phoenix.com.helloaspectj.MainActivity.**(..))")
  public void executeAspectJ() {
  }

  @Before("executeAspectJ()")
  public void beforeAspectJ(JoinPoint joinPoint) throws Throwable {
    Log.d(TAG, "beforeAspectJ: injected -> " + joinPoint.toShortString());
  }
}

第一次接觸AspectJ的看到這段代碼有點摸不著頭腦,解釋一下幾個注解的意思,

  • @Aspect: 告訴ACJ編譯器這是個AspectJ類

  • @PointCut: PointCut是AspectJ中的一個概念,跟它一起的另一個概念是 JoinPoint,這兩個概念一起描述要注入的切面

  • @Before: 表示要注入的位置,常用的有 Before/After/Around,分別表示在執(zhí)行前,執(zhí)行后,和取代原方法

這里@PointCut注解后的參數(shù)表示的意思是對 MainActivity中的所有方法進行注入,參數(shù)用的是正則匹配語法。
下面看看這段代碼執(zhí)行的結(jié)果

07-26 16:04:56.611 22823-22823/? D/AspectTest: beforeAspectJ: injected -> execution(MainActivity.onCreate(..))
07-26 16:04:56.661 22823-22823/? D/AspectTest: beforeAspectJ: injected -> execution(MainActivity.onStart())

看到雖然我們沒有在MainActivity中寫入log打印語句,但是通過AspectJ實現(xiàn)了,在MainActivity兩個生命周期執(zhí)行前插入了我們自己的log。

使用場景

AspectJ只是AOP的其中一種手段,類似的還有用 asm 去修改字節(jié)碼。AOP之所以會有越來越多的人去了解,抽象上來說它可以非常好的去耦合。

高級點的可以用AOP來實現(xiàn)無痕埋點,數(shù)據(jù)收集,甚至修改SDK中動不了的代碼。

以上就是Android中怎么實現(xiàn)切面編程,小編相信有部分知識點可能是我們?nèi)粘9ぷ鲿姷交蛴玫降?。希望你能通過這篇文章學到更多知識。更多詳情敬請關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI