AOP簡介
AOP可能對于廣大開發(fā)者耳熟能詳,它是Aspect Oriented Programming的縮寫匿情,翻譯成中文就是:面向切面編程击奶。這個(gè)可能是面試中經(jīng)常提到的問題匹耕,同時(shí)它也是Spring框架中一個(gè)重大的特性渔呵,AOP主要實(shí)現(xiàn)的目的是針對業(yè)務(wù)處理過程中的切面進(jìn)行提取怒竿,它所面對的是處理過程中的某個(gè)步驟或階段,以獲得邏輯過程中各部分之間低耦合性的隔離效果扩氢,對于我們開發(fā)中最常見的可能就是日志記錄耕驰,事務(wù)處理,異常處理等等录豺。
SpringBoot中使用AOP
接下來介紹SpringBoot項(xiàng)目中如何運(yùn)用AOP朦肘。
新建項(xiàng)目
新建一個(gè)SpringBoot項(xiàng)目,在pom文件中加入AOP依賴双饥,完整代碼如下:
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.dalaoyang</groupId>
<artifactId>springboot_aop</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>jar</packaging>
<name>springboot_aop</name>
<description>springboot_aop</description>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.0.4.RELEASE</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
<java.version>1.8</java.version>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-devtools</artifactId>
<scope>runtime</scope>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>
創(chuàng)建切面
一 直接使用切面
新建一個(gè)日志切面類媒抠,假設(shè)我們需要一個(gè)類來打印進(jìn)入方法或方法執(zhí)行后需要打印的日志。
新建一個(gè)切面類
新建類LogAspect咏花,完整代碼如下:
package com.dalaoyang.aspect;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.springframework.stereotype.Component;
import org.aspectj.lang.annotation.*;
@Aspect
@Component
public class LogAspect {
@Pointcut("execution(public * com.dalaoyang.controller.*.*(..))")
public void LogAspect(){}
@Before("LogAspect()")
public void doBefore(JoinPoint joinPoint){
System.out.println("doBefore");
}
@After("LogAspect()")
public void doAfter(JoinPoint joinPoint){
System.out.println("doAfter");
}
@AfterReturning("LogAspect()")
public void doAfterReturning(JoinPoint joinPoint){
System.out.println("doAfterReturning");
}
@AfterThrowing("LogAspect()")
public void deAfterThrowing(JoinPoint joinPoint){
System.out.println("deAfterThrowing");
}
@Around("LogAspect()")
public Object deAround(ProceedingJoinPoint joinPoint) throws Throwable{
System.out.println("deAround");
return joinPoint.proceed();
}
}
其中:
@Aspect 表明是一個(gè)切面類
@Component 將當(dāng)前類注入到Spring容器內(nèi)
@Pointcut 切入點(diǎn)趴生,其中execution用于使用切面的連接點(diǎn)。使用方法:execution(方法修飾符(可選) 返回類型 方法名 參數(shù) 異常模式(可選)) 迟螺,可以使用通配符匹配字符冲秽,*可以匹配任意字符舍咖。
@Before 在方法前執(zhí)行
@After 在方法后執(zhí)行
@AfterReturning 在方法執(zhí)行后返回一個(gè)結(jié)果后執(zhí)行
@AfterThrowing 在方法執(zhí)行過程中拋出異常的時(shí)候執(zhí)行
@Around 環(huán)繞通知矩父,就是可以在執(zhí)行前后都使用,這個(gè)方法參數(shù)必須為ProceedingJoinPoint排霉,proceed()方法就是被切面的方法窍株,上面四個(gè)方法可以使用JoinPoint,JoinPoint包含了類名,被切面的方法名球订,參數(shù)等信息后裸。
二 利用自定義注解使用AOP
新建自定義注解
新建自定義注解,新建注解與新建接口類似冒滩,將interface改為@interface即可微驶。
package com.dalaoyang.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface DoneTime {
String param() default "";
}
創(chuàng)建自定義注解對應(yīng)切面
創(chuàng)建自定義注解對應(yīng)切面,與上一中情況的切面類似开睡,這里就不在贅述了因苹,代碼如下:
package com.dalaoyang.aspect;
import com.dalaoyang.annotation.DoneTime;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
import java.util.Date;
@Aspect
@Component
public class DoneTimeAspect {
@Around("@annotation(doneTime)")
public Object around(ProceedingJoinPoint joinPoint, DoneTime doneTime) throws Throwable {
System.out.println("方法開始時(shí)間是:"+new Date());
Object o = joinPoint.proceed();
System.out.println("方法結(jié)束時(shí)間是:"+new Date()) ;
return o;
}
}
創(chuàng)建Controller測試
創(chuàng)建一個(gè)IndexController進(jìn)行測試,其實(shí)就是兩個(gè)普通的Web請求方法篇恒,其中index使用了自定義注解扶檐,index2沒有使用自定義注解,完整代碼如下:
package com.dalaoyang.controller;
import com.dalaoyang.annotation.DoneTime;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class IndexController {
@GetMapping("/index")
@DoneTime(param = "IndexController")
public String index(){
System.out.println("方法執(zhí)行");
return "hello dalaoyang";
}
@GetMapping("/index2")
public String index2(){
System.out.println("方法2執(zhí)行");
return "hello dalaoyang";
}
}
總結(jié)
本文只是簡單介紹一下SpringBoot使用Spring AOP胁艰,當(dāng)然它可能還有其他使用方法款筑,具體的情況需要根據(jù)我們實(shí)際情況來使用。
Java高架構(gòu)師腾么、分布式架構(gòu)奈梳、高可擴(kuò)展、高性能解虱、高并發(fā)颈嚼、性能優(yōu)化、Spring boot饭寺、Redis阻课、ActiveMQ、Nginx艰匙、Mycat限煞、Netty、Jvm大型分布式項(xiàng)目實(shí)戰(zhàn)學(xué)習(xí)架構(gòu)師視頻免費(fèi)獲取架構(gòu)群:854180697
群鏈接:加群鏈接
寫在最后:歡迎留言討論员凝,加關(guān)注署驻,持續(xù)更新!