AOP面向切面编程在Android中的使用
GitHub地址(歡迎下載完整Demo)
https://github.com/ganchuanpu/AOPDemo
項目需求描述
我想類似于這樣的個人中心的界面,大家都不會陌生吧。那幾個有箭頭的地方都是可以點擊進行頁面跳轉的,但是需要先判斷用戶是否登錄,如果已經登錄,則正常跳轉,如果沒有登錄,則跳轉到登錄頁面先登錄,但凡是有注冊,登錄的APP,這樣的操作,大家應該都很熟悉吧。一般情況下,我們的邏輯是這樣的…
/*** 跳轉到我的關注頁面 */ public void toMyAttention() {// 判斷當前用戶是否登錄if(LoginHelper.isLogin(this)) {// 如果登錄才跳轉,進入我的關注頁面Intent intent = new Intent(this, WaitReceivingActivity.class);startActivity(intent);}else{//跳轉到登錄頁面,先登錄Intent intent = new Intent(this, LoginActivity.class);startActivity(intent);} }重復的體力勞動,想想都可怕。而且類似的還有網絡判斷,權限管理,Log日志的統一管理這樣的問題。那么,我們也沒有更優雅的方式來解決這一類的問題呢,答案是有的。
先給出我解決了上述問題之后的代碼
/*** 跳轉到我的關注頁面*/@CheckLoginpublic void toMyAttention() {Intent intent = new Intent(this, WaitReceivingActivity.class);startActivity(intent);}AspectJ
AspectJ實際上是對AOP編程思想的一個實踐,AOP雖然是一種思想,但就好像OOP中的Java一樣,一些先行者也開發了一套語言來支持AOP。目前用得比較火的就是AspectJ了,它是一種幾乎和Java完全一樣的語言,而且完全兼容Java(AspectJ應該就是一種擴展Java,但它不是像Groovy那樣的拓展。)。當然,除了使用AspectJ特殊的語言外,AspectJ還支持原生的Java,只要加上對應的AspectJ注解就好。所以,使用AspectJ有兩種方法:
- 完全使用AspectJ的語言。這語言一點也不難,和Java幾乎一樣,也能在AspectJ中調用Java的任何類庫。AspectJ只是多了一些關鍵詞罷了。
- 或者使用純Java語言開發,然后使用AspectJ注解,簡稱@AspectJ。
基礎概念
- Aspect 切面:切面是切入點和通知的集合。
PointCut 切入點:切入點是指那些通過使用一些特定的表達式過濾出來的想要切入Advice的連接點。
Advice 通知:通知是向切點中注入的代碼實現方法。
Joint Point 連接點:所有的目標方法都是連接點.
- Weaving 編織:主要是在編譯期使用AJC將切面的代碼注入到目標中, 并生成出代碼混合過的.class的過程.
實踐步驟
1、在android studio中直接配置AspectJ,這個配置很重要,如果失敗,后面就無法成功,先貼出我的配置,在app的build.gradle中做如下配置
1 apply plugin: 'com.android.application' 2 import org.aspectj.bridge.IMessage 3 import org.aspectj.bridge.MessageHandler 4 import org.aspectj.tools.ajc.Main 5 6 buildscript { 7 repositories { 8 mavenCentral() 9 } 10 dependencies { 11 classpath 'org.aspectj:aspectjtools:1.8.9' 12 classpath 'org.aspectj:aspectjweaver:1.8.9' 13 } 14 } 15 repositories { 16 mavenCentral() 17 } 18 final def log = project.logger 19 final def variants = project.android.applicationVariants 20 variants.all { variant -> 21 if (!variant.buildType.isDebuggable()) { 22 log.debug("Skipping non-debuggable build type '${variant.buildType.name}'.") 23 return; 24 } 25 26 JavaCompile javaCompile = variant.javaCompile 27 javaCompile.doLast { 28 String[] args = ["-showWeaveInfo", 29 "-1.8", 30 "-inpath", javaCompile.destinationDir.toString(), 31 "-aspectpath", javaCompile.classpath.asPath, 32 "-d", javaCompile.destinationDir.toString(), 33 "-classpath", javaCompile.classpath.asPath, 34 "-bootclasspath", project.android.bootClasspath.join(File.pathSeparator)] 35 log.debug "ajc args: " + Arrays.toString(args) 36 37 MessageHandler handler = new MessageHandler(true); 38 new Main().run(args, handler); 39 for (IMessage message : handler.getMessages(null, true)) { 40 switch (message.getKind()) { 41 case IMessage.ABORT: 42 case IMessage.ERROR: 43 case IMessage.FAIL: 44 log.error message.message, message.thrown 45 break; 46 case IMessage.WARNING: 47 log.warn message.message, message.thrown 48 break; 49 case IMessage.INFO: 50 log.info message.message, message.thrown 51 break; 52 case IMessage.DEBUG: 53 log.debug message.message, message.thrown 54 break; 55 } 56 } 57 } 58 } 59 60 android { 61 compileSdkVersion 25 62 buildToolsVersion "25.0.2" 63 defaultConfig { 64 applicationId "com.zx.aopdemo" 65 minSdkVersion 17 66 targetSdkVersion 25 67 versionCode 1 68 versionName "1.0" 69 testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" 70 } 71 buildTypes { 72 release { 73 minifyEnabled false 74 proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' 75 } 76 } 77 } 78 79 dependencies { 80 compile fileTree(dir: 'libs', include: ['*.jar']) 81 androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', { 82 exclude group: 'com.android.support', module: 'support-annotations' 83 }) 84 compile 'com.android.support:appcompat-v7:25.3.1' 85 compile 'com.android.support.constraint:constraint-layout:1.0.2' 86 compile 'org.aspectj:aspectjrt:1.8.9' 87 testCompile 'junit:junit:4.12' 88 }為什么這么配置?因為AspectJ是對java的擴展,而且是完全兼容java的。但是編譯時得用Aspect專門的編譯器,這里的配置就是使用Aspect的編譯器,單獨加入aspectj依賴是不行的。到這里準備工作已完成,可以開始看看具體實現了。
2、創建切面AspectJ
用來處理觸發切面的回調
這里要使用Aspect的編譯器編譯必須給類打上標注,@Aspect。
還有這里的Pointcut注解,就是切點,即觸發該類的條件。里面的字符串如下
在Pointcut這里,我使用了execution,也就是以方法執行時為切點,觸發Aspect類。而execution里面的字符串是觸發條件,也是具體的切點。我來解釋一下參數的構成。“execution(@com.zx.aopdemo.login.CheckLogin * *(..))”這個條件是所有加了CheckLogin注解的方法或屬性都會是切點,范圍比較廣。
- **:表示是任意包名
- ..:表示任意類型任意多個參數
“com.zx.aopdemo.login.CheckLogin”這是我的項目包名下需要指定類的絕對路徑。再來看看@Around,Around是指JPoint執行前或執行后被觸發,除了Around還有其他幾種方式。
創建完Aspect類之后,還需要一個注解類,它的作用是:哪里需要做切點,那么哪里就用注解標注一下,這樣方便快捷。
3、創建注解類
package com.zx.aopdemo.login;import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target;@Target(ElementType.METHOD) //可以注解在方法 上 @Retention(RetentionPolicy.RUNTIME) //運行時(執行時)存在 public @interface CheckLogin { }4、Activity使用登錄的注解
public class LoginActivity extends AppCompatActivity implements View.OnClickListener, RadioGroup.OnCheckedChangeListener {private RadioGroup radioGroup;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_login);test();}@CheckLoginpublic void test(){ Log.i("tag","判斷是否登錄"); }test()方法執行時就是一個切點。在執行test()時,會回調上面的CheckLoginAspectJ類的executionCheckLogin()方法。然后會執行
如下方法
如果使用的是以方法相關為切點,那么使用MethodSignature來接收joinPoint的Signature。如果是屬性或其他的,那么可以使用Signature類來接收。之后可以使用Signature來獲取注解類。,那么通過jointPoint.getThis()獲取使用該注解的的上下文對象
https://www.cnblogs.com/ganchuanpu/p/8594877.html
總結
以上是生活随笔為你收集整理的AOP面向切面编程在Android中的使用的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 时间序列相关算法与分析步骤
- 下一篇: Android热修复技术原理详解(最新最