溫馨提示×

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

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何在Android中實(shí)現(xiàn)切面編程

發(fā)布時(shí)間:2021-06-08 16:47:07 來源:億速云 閱讀:166 作者:Leah 欄目:移動(dòng)開發(fā)

本篇文章為大家展示了如何在Android中實(shí)現(xiàn)切面編程,內(nèi)容簡(jiǎn)明扼要并且容易理解,絕對(duì)能使你眼前一亮,通過這篇文章的詳細(xì)介紹希望你能有所收獲。

安裝AspectJ

Android上的ApsectJ開發(fā)由幾部分組成,AspectJ gradle插件,ApsectJ依賴,還有 AspectJ編譯器。

首先安裝AspectJ編譯器很簡(jiǎn)單,就跟安裝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)境的話直接運(yùn)行這個(gè)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

配置完后運(yùn)行 ajc -v 應(yīng)該可以看到對(duì)應(yīng)輸出

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
  }
}

然后在項(xiàng)目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編譯完成后追加一個(gè) acj 的編譯流程,

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

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

implementation 'org.aspectj:aspectjrt:1.9.0'

創(chuàng)建AspectJ代碼

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

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

以一個(gè)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)在我們要寫一個(gè)AspectJ類,這個(gè)類看起來會(huì)跟一般的Java類有點(diǎn)不同,可以理解為它只是用注解作為媒介,讓ACJ編譯器知道要去注入哪些方法。

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

@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的看到這段代碼有點(diǎn)摸不著頭腦,解釋一下幾個(gè)注解的意思,

  • @Aspect: 告訴ACJ編譯器這是個(gè)AspectJ類

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

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

這里@PointCut注解后的參數(shù)表示的意思是對(duì) MainActivity中的所有方法進(jìn)行注入,參數(shù)用的是正則匹配語(yǔ)法。

下面看看這段代碼執(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打印語(yǔ)句,但是通過AspectJ實(shí)現(xiàn)了,在MainActivity兩個(gè)生命周期執(zhí)行前插入了我們自己的log。

上述內(nèi)容就是如何在Android中實(shí)現(xiàn)切面編程,你們學(xué)到知識(shí)或技能了嗎?如果還想學(xué)到更多技能或者豐富自己的知識(shí)儲(chǔ)備,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

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

AI