欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java SpringBoot實現(xiàn)AOP

 更新時間:2021年09月16日 09:29:41   作者:Tttori  
AOP包括連接點(JoinPoint)、切入點(Pointcut)、增強(Advisor)、切面(Aspect)、AOP代理(AOP Proxy),具體的方法和類型下面文章會舉例說明,感興趣的小伙伴和小編一起閱讀全文吧

1、AOP基本總結(jié)

連接點JoinPoint):

連接點是程序運行的某個階段點,如方法調(diào)用、異常拋出等

切入點Pointcut):

切入點是JoinPoint的集合
是程序中需要注入Advice的位置的集合,即Advice在什么條件下才能被觸發(fā)

增強Advisor):

增強是切入點PointcutAdvice的綜合體,即在連接點JoinPoint上執(zhí)行的行為
通過JDK/CGLIB代理模式實現(xiàn)AOP

切面Aspect):

@Aspect通常是一個類的注解,通常與@Component搭配使用

AOP代理AOP Proxy):

AOP使用動態(tài)代理模式創(chuàng)建對象,從而實現(xiàn)在連接點JoinPoint處插入增強
其中JDK只能代理接口,CGLIB基于子類但不能代理final類

2、常用方法

3、增強類型

  • @Before:前置增強,在某個JoinPoint執(zhí)行前的增強
  • @After:final增強,不管拋異常還是正常退出都執(zhí)行的增強
  • @AfterReturning:后置增強,方法正常退出時執(zhí)行
  • @AfterThrowing:異常拋出增強,拋出異常后執(zhí)行
  • @Around:環(huán)繞增強,包圍一個連接點的增強,最強大的一個方式,且常用

4、示例說明

學了一下AOP的使用,寫了個@Arounddemo,將幾個查詢操作存入數(shù)據(jù)庫作為Log并且定時清理過期數(shù)據(jù)

本人的Demo用的是Dubbo框架,而AOP的示例寫在了Provider中,大概結(jié)構(gòu)如下:

monitor:

  • annotation:注解類
  • aop:切面的定義及實現(xiàn)
  • impl:UserAopTask接口的實現(xiàn)類

 1)UserLog實體類

@Data
@ToString
@NoArgsConstructor
@AllArgsConstructor
public class UserLog implements Serializable {
    private Integer id;
    private String methodName;
    private String methodArgs;
    private String classFullName;
    private String className;
    private Date invokeTime;
    private Double costTime;
}
 

2)LogTaskMapper對應mapper接口

public interface LogTaskMapper {
    /**
     * TEST AOP INSERT INFO INTO TABLE
     * @param userLog
     */
    void insertUserLog(UserLog userLog);

    /**
     * DELETE LOGS IN TABLE LAST x MINUTES
     * @param minutes
     */
    void deleteUserLog(int minutes);
}
 

3)UserAopTask接口

public interface UserAopTask {
    void insertUserLog(UserLog log);
}
 

4)UserAopTaskImpl實現(xiàn)類

@Component
public class UserAopTaskImpl implements UserAopTask {

    private static final Logger logger = LoggerFactory.getLogger(UserAopTask.class);

    @Autowired
    private LogTaskMapper logTaskMapper;

    private ExecutorService logHandler = Executors.newFixedThreadPool(1);//采用線程池復用一個線程執(zhí)行

    private static final int MINUTES_LOG_RETAIN = 30;//數(shù)據(jù)庫中數(shù)據(jù)保留時間


    @Override
    public void insertUserLog(UserLog log) {
        logHandler.submit(new logSubmitTask(log));
    }

    //內(nèi)部類
    class logSubmitTask implements Runnable{

        private UserLog userLog;

        public logSubmitTask(UserLog userLog){
            this.userLog = userLog;
        }

        @Override
        public void run() {
            logTaskMapper.insertUserLog(userLog);
        }
    }

    //定時清理任務
    @Scheduled(cron = "0 30 * * * *")
    public void scheduledDeleteLog(){
        logger.info("開始清除[{}]分鐘之前的圖表查詢?nèi)罩?..", MINUTES_LOG_RETAIN);
        logTaskMapper.deleteUserLog(-1 * MINUTES_LOG_RETAIN);
    }

}
 

5)TestUserAop切面類

@Aspect//切面
@Component//Spring容器管理
public class TestUserAop {
    private static final Logger logger = LoggerFactory.getLogger(TestUserAop.class);

    @Autowired
    private UserAopTask userAopTask;
   //使用環(huán)繞增強,第一參數(shù)必須是ProceedingJoinPoint
    @Around(value = "@annotation(annotation)")//和注解類參數(shù)名保持一致
    public Object aroundUserInfo(ProceedingJoinPoint pjp, TestUserAnnotation annotation) throws Throwable{
        UserLog userLog = new UserLog();

        System.out.println("=====================ANNOTATION BEGIN=====================");

        Date date = new Date();
        Long methodStart = date.getTime();//timestamp
        System.out.println("ANNOTATION 開始耗時統(tǒng)計: "+ date);

        userLog.setInvokeTime(date);

        Object[] argsObj = pjp.getArgs();
        Object res = pjp.proceed(argsObj);//利用反射調(diào)用目標方法

        Long methodCost = System.currentTimeMillis() - methodStart;
        double cost = methodCost/1000d;//timestamp 轉(zhuǎn)換為 seconds

        System.out.println("ANNOTATION 調(diào)用方法總耗時: "+ String.format("%.3f",cost) +" s");//保留3位小數(shù)
        System.out.println("ANNOTATION 調(diào)用方法: "+annotation.methodName());//目標方法
        System.out.println("ANNOTATION 調(diào)用方法參數(shù): "+ new Integer((Integer) argsObj[0]));//我的參數(shù)就1個或者無參
        System.out.println("ANNOTATION 調(diào)用類: "+pjp.getSignature().getDeclaringTypeName());//全類名
        System.out.println("ANNOTATION 調(diào)用類名: "+pjp.getSignature().getDeclaringType().getSimpleName());//類名
        System.out.println("ANNOTATION 調(diào)用結(jié)果: "+ JSON.toJSON(res));
        System.out.println("=====================ANNOTATION FINISHED=====================");

        userLog.setCostTime(Double.parseDouble(String.format("%.3f",cost)));
        userLog.setClassFullName(pjp.getSignature().getDeclaringTypeName());
        userLog.setClassName(pjp.getSignature().getDeclaringType().getSimpleName());
        userLog.setMethodName(annotation.methodName());
        userLog.setMethodArgs(Integer.toString(new Integer((Integer) argsObj[0])));

        userAopTask.insertUserLog(userLog);

        return res;

    }
}
 

6)TestUserAnnotation注解類

我在service層寫的AOP demo,對目標方法使用注解,注解名為注解類名即可,如@TestUserAnnotation

@Retention(RetentionPolicy.RUNTIME)//運行時有效
@Target(ElementType.METHOD)//作用于方法
@Documented
public @interface TestUserAnnotation {
    String methodName() default "";//方法名,默認為空字符串
}
 

7)LogTaskMapper.xml

最后貼個代碼,為上面提到的定時任務,用到的是date_add()方法,其中的 "&lt;" 意為 "<"

<delete id="deleteUserLog" parameterType="java.lang.Integer">
        delete from invoke_log
        where invoke_time &lt; date_add(current_timestamp,interval #{minutes} minute)
</delete>
 

5、結(jié)果展示

演示一下AOP的效果,將@TestUserAnnotation注解在方法getUserInfo(),即獲取用戶信息

Demo中利用AOP的@Around環(huán)繞增強,實現(xiàn)了統(tǒng)計方法調(diào)用運行消耗時間,以及統(tǒng)計調(diào)用方法名、類名等信息:

 

 調(diào)用方法getUserInfo后的統(tǒng)計結(jié)果:

到此這篇關于Java SpringBoot實現(xiàn)AOP 的文章就介紹到這了,更多相關SpringBoot實現(xiàn)AOP 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • idea文件上有鎖文件只讀不可編輯的解決

    idea文件上有鎖文件只讀不可編輯的解決

    這篇文章主要介紹了idea文件上有鎖文件只讀不可編輯的解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • IKAnalyzer結(jié)合Lucene實現(xiàn)中文分詞(示例講解)

    IKAnalyzer結(jié)合Lucene實現(xiàn)中文分詞(示例講解)

    下面小編就為大家?guī)硪黄狪KAnalyzer結(jié)合Lucene實現(xiàn)中文分詞(示例講解)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-10-10
  • Java數(shù)組拷貝的幾種方法小結(jié)

    Java數(shù)組拷貝的幾種方法小結(jié)

    本文主要介紹了Java數(shù)組拷貝的幾種方法小結(jié),包括for循環(huán)、System.arraycopy()、Arrays.copyOf()、Arrays.copyOfRange()和clone()方法,具有一定的參考價值,感興趣的可以了解一下
    2023-09-09
  • java獲取新insert數(shù)據(jù)自增id的實現(xiàn)方法

    java獲取新insert數(shù)據(jù)自增id的實現(xiàn)方法

    這篇文章主要介紹了java獲取新insert數(shù)據(jù)自增id的實現(xiàn)方法,在具體生成id的時候,我們的操作順序一般是:先在主表中插入記錄,然后獲得自動生成的id,以它為基礎插入從表的記錄,需要的朋友可以參考下
    2019-06-06
  • Java判斷字節(jié)流是否是 UTF8編碼方法示例

    Java判斷字節(jié)流是否是 UTF8編碼方法示例

    這篇文章主要我大家介紹了Java判斷字節(jié)流是否是 UTF8編碼方法示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-07-07
  • JAVA各種加密與解密方式總結(jié)大全

    JAVA各種加密與解密方式總結(jié)大全

    這篇文章主要給大家介紹了關于JAVA各種加密與解密方式總結(jié)的相關資料,加密是指對原來為明文的文件或數(shù)據(jù)按某種算法進行處理,使其成為不可讀的一段代碼,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2023-07-07
  • mybatis學習筆記之mybatis注解配置詳解

    mybatis學習筆記之mybatis注解配置詳解

    本篇文章主要介紹了mybatis學習筆記之mybatis注解配置詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-12-12
  • JAVA中通過Hibernate-Validation進行參數(shù)驗證

    JAVA中通過Hibernate-Validation進行參數(shù)驗證

    這篇文章主要介紹了JAVA中通過Hibernate-Validation進行參數(shù)驗證,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-04-04
  • java如何將實體類轉(zhuǎn)換成json并在控制臺輸出

    java如何將實體類轉(zhuǎn)換成json并在控制臺輸出

    這篇文章主要介紹了java如何將實體類轉(zhuǎn)換成json并在控制臺輸出問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-11-11
  • Json字符串轉(zhuǎn)Java對象和List代碼實例

    Json字符串轉(zhuǎn)Java對象和List代碼實例

    這篇文章主要介紹了Json字符串轉(zhuǎn)Java對象和List代碼實例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-06-06

最新評論