国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統(tǒng) > Android > 正文

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

2019-10-21 21:44:56
字體:
供稿:網(wǎng)友

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

先說熟悉的面向?qū)ο缶幊?OOP,通常都是用各種對(duì)象/模塊來負(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)槎冀尤肓耍运麄冊(cè)谀撤N程度上就有了耦合,要修改Log模塊的實(shí)現(xiàn)的時(shí)候會(huì)影響到其他三個(gè)模塊的實(shí)現(xiàn)。

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

(其實(shí)這是一篇AspectJ環(huán)境配置指南)

安裝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.MessageHandlerimport org.aspectj.tools.ajc.Mainbuildscript {  repositories {    mavenCentral()  }  dependencies {    classpath 'org.aspectj:aspectjtools:1.8.9'    classpath 'org.aspectj:aspectjweaver:1.8.9'  }}repositories {  mavenCentral()}final def log = project.loggerfinal def variants = project.android.applicationVariantsvariants.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ì)一臉懵逼,不過目前先不用管具體的語法含義,

先跑起來環(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è)方法,

@Aspectpublic 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ù)用的是正則匹配語法。

下面看看這段代碼執(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。

使用場(chǎng)景

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

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

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


注:相關(guān)教程知識(shí)閱讀請(qǐng)移步到Android開發(fā)頻道。
發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 高密市| 邯郸市| 石狮市| 吉安市| 景泰县| 汾阳市| 凤城市| 湟源县| 宜君县| 灵武市| 那坡县| 巢湖市| 龙里县| 石家庄市| 西安市| 垣曲县| 虎林市| 靖西县| 南阳市| 柳州市| 德阳市| 冕宁县| 丰都县| 乡城县| 阿拉尔市| 苗栗县| 岳池县| 烟台市| 垣曲县| 印江| 芒康县| 汾阳市| 上饶市| 蒙山县| 曲水县| 岑巩县| 泰顺县| 普陀区| 抚顺县| 泗洪县| 文化|