亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Android切面編程知識(shí)點(diǎn)詳解

 更新時(shí)間:2018年07月29日 10:00:39   投稿:laozhang  
這篇文章給大家整理了關(guān)于Android進(jìn)階資深開發(fā)必備技能-切面編程的相關(guān)知識(shí)點(diǎn)內(nèi)容,有興趣的朋友可以參考學(xué)習(xí)下。

切面編程聽起來可能有點(diǎn)陌生,不過現(xiàn)在越來越多的開發(fā)團(tuán)隊(duì)正在用上這種技術(shù)。

先說熟悉的面向?qū)ο缶幊?OOP,通常都是用各種對象/模塊來負(fù)責(zé)具體的功能,互相之間盡量不耦合。

切面編程AOP(aspect-priented programming)是為了解決OOP中耦合無法解除的問題而產(chǎn)生的。

打個(gè)比方現(xiàn)在項(xiàng)目中有負(fù)責(zé)網(wǎng)絡(luò)/數(shù)據(jù)存儲(chǔ)/UI幾個(gè)模塊,每個(gè)模塊都接入了另外一個(gè)Log模塊。

雖然Log不屬于前面三個(gè)的功能,但因?yàn)槎冀尤肓?,所以他們在某種程度上就有了耦合,要修改Log模塊的實(shí)現(xiàn)的時(shí)候會(huì)影響到其他三個(gè)模塊的實(shí)現(xiàn)。

這篇文章用最簡單的例子來描述AOP是怎么解決這種問題的。

(其實(shí)這是一篇AspectJ環(huá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)境的話直接運(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)該可以看到對應(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中的對象,用來接收參數(shù)然后進(jìn)行 acj 編譯的。

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

implementation 'org.aspectj:aspectjrt:1.9.0'

創(chuàng)建AspectJ代碼

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

先跑起來環(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ù)表示的意思是對 MainActivity中的所有方法進(jìn)行注入,參數(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實(shí)現(xiàn)了,在MainActivity兩個(gè)生命周期執(zhí)行前插入了我們自己的log。

使用場景

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

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

上面的整個(gè)DEMO代碼可以從GitHub上獲取,后臺(tái)回復(fù)"切面"就可以獲取下載鏈接。

相關(guān)文章

最新評論