本篇文章給大家分享的是有關(guān)Android中怎么實(shí)現(xiàn)切面編程,小編覺(jué)得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話(huà)不多說(shuō),跟著小編一起來(lái)看看吧。
神池ssl適用于網(wǎng)站、小程序/APP、API接口等需要進(jìn)行數(shù)據(jù)傳輸應(yīng)用場(chǎng)景,ssl證書(shū)未來(lái)市場(chǎng)廣闊!成為創(chuàng)新互聯(lián)公司的ssl證書(shū)銷(xiāo)售渠道,可以享受市場(chǎng)價(jià)格4-6折優(yōu)惠!如果有意向歡迎電話(huà)聯(lián)系或者加微信:18980820575(備注:SSL證書(shū)合作)期待與您的合作!安裝AspectJ
Android上的ApsectJ開(kāi)發(fā)由幾部分組成,AspectJ gradle插件,ApsectJ依賴(lài),還有 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)境的話(huà)直接運(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依賴(lài)
構(gòu)建帶AspectJ支持的Android App的流程是先按正常流程編譯出 .class 文件后,再用 ajc 編譯器在 .class文件中插入我們需要的代碼。
首先需要把 AspectJ 依賴(lài)加到 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ì)象,用來(lái)接收參數(shù)然后進(jìn)行 acj 編譯的。
最后再把 dependencies依賴(lài)加上對(duì)AspectJ的支持就可以了,
implementation 'org.aspectj:aspectjrt:1.9.0'
創(chuàng)建AspectJ代碼
下面這部分代碼看起來(lái)會(huì)一臉懵逼,不過(guò)目前先不用管具體的語(yǔ)法含義,
先跑起來(lái)環(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)在我們要寫(xiě)一個(gè)AspectJ類(lèi),這個(gè)類(lèi)看起來(lái)會(huì)跟一般的Java類(lèi)有點(diǎn)不同,可以理解為它只是用注解作為媒介,讓ACJ編譯器知道要去注入哪些方法。
這個(gè)類(lèi)要做的事情是告訴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類(lèi)
@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())
看到雖然我們沒(méi)有在MainActivity中寫(xiě)入log打印語(yǔ)句,但是通過(guò)AspectJ實(shí)現(xiàn)了,在MainActivity兩個(gè)生命周期執(zhí)行前插入了我們自己的log。
使用場(chǎng)景
AspectJ只是AOP的其中一種手段,類(lèi)似的還有用 asm 去修改字節(jié)碼。AOP之所以會(huì)有越來(lái)越多的人去了解,抽象上來(lái)說(shuō)它可以非常好的去耦合。
高級(jí)點(diǎn)的可以用AOP來(lái)實(shí)現(xiàn)無(wú)痕埋點(diǎn),數(shù)據(jù)收集,甚至修改SDK中動(dòng)不了的代碼。
以上就是Android中怎么實(shí)現(xiàn)切面編程,小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見(jiàn)到或用到的。希望你能通過(guò)這篇文章學(xué)到更多知識(shí)。更多詳情敬請(qǐng)關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。
當(dāng)前標(biāo)題:Android中怎么實(shí)現(xiàn)切面編程-創(chuàng)新互聯(lián)
文章網(wǎng)址:http://www.rwnh.cn/article14/cchpde.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供商城網(wǎng)站、品牌網(wǎng)站設(shè)計(jì)、網(wǎng)站維護(hù)、網(wǎng)站設(shè)計(jì)公司、定制網(wǎng)站、品牌網(wǎng)站建設(shè)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶(hù)投稿、用戶(hù)轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話(huà):028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容