AOP面向切麵編程在Android中的使用

来源:https://www.cnblogs.com/ganchuanpu/archive/2018/03/18/8594877.html
-Advertisement-
Play Games

GitHub地址(歡迎下載完整Demo) https://github.com/ganchuanpu/AOPDemo 項目需求描述 我想類似於這樣的個人中心的界面,大家都不會陌生吧。那幾個有箭頭的地方都是可以點擊進行頁面跳轉的,但是需要先判斷用戶是否登錄,如果已經登錄,則正常跳轉,如果沒有登錄,則跳 ...


GitHub地址(歡迎下載完整Demo)

https://github.com/ganchuanpu/AOPDemo

項目需求描述

個人中心.jpg

我想類似於這樣的個人中心的界面,大家都不會陌生吧。那幾個有箭頭的地方都是可以點擊進行頁面跳轉的,但是需要先判斷用戶是否登錄,如果已經登錄,則正常跳轉,如果沒有登錄,則跳轉到登錄頁面先登錄,但凡是有註冊,登錄的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日誌的統一管理這樣的問題。那麼,我們也沒有更優雅的方式來解決這一類的問題呢,答案是有的。

先給出我解決了上述問題之後的代碼

/**
    *  跳轉到我的關註頁面
    */
    @CheckLogin
    public 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
用來處理觸發切麵的回調

 1 @Aspect
 2 public class CheckLoginAspectJ {
 3     private static final String TAG = "CheckLogin";
 4 
 5     /**
 6      * 找到處理的切點
 7      * * *(..)  可以處理CheckLogin這個類所有的方法
 8      */
 9     @Pointcut("execution(@com.zx.aopdemo.login.CheckLogin  * *(..))")
10     public void executionCheckLogin() {
11     }
12 
13     /**
14      * 處理切麵
15      *
16      * @param joinPoint
17      * @return
18      */
19     @Around("executionCheckLogin()")
20     public Object checkLogin(ProceedingJoinPoint joinPoint) throws Throwable {
21         Log.i(TAG, "checkLogin: ");
22         MethodSignature signature = (MethodSignature) joinPoint.getSignature();
23         CheckLogin checkLogin = signature.getMethod().getAnnotation(CheckLogin.class);
24         if (checkLogin != null) {
25             Context context = (Context) joinPoint.getThis();
26             if (MyApplication.isLogin) {
27                 Log.i(TAG, "checkLogin: 登錄成功 ");
28                 return joinPoint.proceed();
29             } else {
30                 Log.i(TAG, "checkLogin: 請登錄");
31                 Toast.makeText(context, "請登錄", Toast.LENGTH_SHORT).show();
32                 return null;
33             }
34         }
35         return joinPoint.proceed();
36     }
37 }

這裡要使用Aspect的編譯器編譯必須給類打上標註,@Aspect。
還有這裡的Pointcut註解,就是切點,即觸發該類的條件。裡面的字元串如下

AspectJ中的Join Point.png

在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;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_login);

       test();
    }

    @CheckLogin
   public void test(){  
        Log.i("tag","判斷是否登錄");  
    }  

test()方法執行時就是一個切點。在執行test()時,會回調上面的CheckLoginAspectJ類的executionCheckLogin()方法。然後會執行
如下方法

 1 /**
 2      * 處理切麵
 3      *
 4      * @param joinPoint
 5      * @return
 6      */
 7     @Around("executionCheckLogin()")
 8     public Object checkLogin(ProceedingJoinPoint joinPoint) throws Throwable {
 9         Log.i(TAG, "checkLogin: ");
10         MethodSignature signature = (MethodSignature) joinPoint.getSignature();
11         CheckLogin checkLogin = signature.getMethod().getAnnotation(CheckLogin.class);
12         if (checkLogin != null) {
13             Context context = (Context) joinPoint.getThis();
14             if (MyApplication.isLogin) {
15                 Log.i(TAG, "checkLogin: 登錄成功 ");
16                 return joinPoint.proceed();
17             } else {
18                 Log.i(TAG, "checkLogin: 請登錄");
19                 Toast.makeText(context, "請登錄", Toast.LENGTH_SHORT).show();
20                 return null;
21             }
22         }
23         return joinPoint.proceed();
24     }

如果使用的是以方法相關為切點,那麼使用MethodSignature來接收joinPoint的Signature。如果是屬性或其他的,那麼可以使用Signature類來接收。之後可以使用Signature來獲取註解類。,那麼通過jointPoint.getThis()獲取使用該註解的的上下文對象


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 閱讀目錄 常用 搜索、導航 編寫代碼 重構 一、常用 代碼保存 描述:該操作可以用於在書寫代碼的過程中進行快速保存 調用:菜單欄 File -> Save All 快捷鍵:Ctrl + S 最大化/最小化代碼編輯視窗 描述:隱藏所有工具視窗,代碼編輯視窗會最大化/最小化 調用:菜單欄 Window ...
  • 簡要:本系列文章講會對expo進行全面的介紹,本人從2017年6月份接觸expo以來,對expo的研究斷斷續續,一路走來將近10個月,廢話不多說,接下來你看到內容,講全部來與官網 我猜去全部機翻+個人修改補充+demo測試的形式,對expo進行一次大補血!歡迎加入expo興趣學習交流群:597732 ...
  • 前言: 最近想要在酷安網上傳apk,註冊開發者的時候需要申請驗證,驗證需要兩個apk,一個是自己的apk(需要簽名),另外一個則是下載酷安的模板生成的一個簽名包(使用的簽名要與之前的簽名自己的apk一樣),在簽名包的生成之中,gradle報了幾條錯誤,之後經過漫長的百度,總算是完美解決,便是把這個記 ...
  • Github地址(完整Demo,歡迎下載) https://github.com/ganchuanpu/ItemGroup 效果圖 attrs.xml 獲取到各屬性 xml佈局文件 調用的activity ...
  • 簡要:本系列文章講會對expo進行全面的介紹,本人從2017年6月份接觸expo以來,對expo的研究斷斷續續,一路走來將近10個月,廢話不多說,接下來你看到內容,講全部來與官網 我猜去全部機翻+個人修改補充+demo測試的形式,對expo進行一次大補血!歡迎加入expo興趣學習交流群:597732 ...
  • 簡要:本系列文章講會對expo進行全面的介紹,本人從2017年6月份接觸expo以來,對expo的研究斷斷續續,一路走來將近10個月,廢話不多說,接下來你看到內容,講全部來與官網 我猜去全部機翻+個人修改補充+demo測試的形式,對expo進行一次大補血!歡迎加入expo興趣學習交流群:597732 ...
  • 簡要:本系列文章講會對expo進行全面的介紹,本人從2017年6月份接觸expo以來,對expo的研究斷斷續續,一路走來將近10個月,廢話不多說,接下來你看到內容,講全部來與官網 我猜去全部機翻+個人修改補充+demo測試的形式,對expo進行一次大補血!歡迎加入expo興趣學習交流群:597732 ...
  • 前置環境 需要JAVA環境,如果沒有配置JAVA 環境,點下麵鏈接按教程進行配置: Java開發環境搭建:http://www.cnblogs.com/magisk/p/8496396.html 所需要的工具 1、apktool,功能:反編譯出apk所需要的資源文件和佈局設置文件等, 下載地址:ht ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...