對某個類型中的方法進行攔截,然后加入固定的業務邏輯,這是AOP面向切面編程可以做的事,在springboot里實現aop的方法也有很多,spring-boot-starter-aop
或者aspectjweaver
都是可以實現的,不過我們在實現之前,先來看一下aop里的幾個概念。
概念
-
切面(Aspect):是指橫切多個對象的關注點的一個模塊化,事務管理就是J2EE應用中橫切關注點的很好示例。在Spring AOP中,切面通過常規類(基本模式方法)或者通過使用了注解@Aspect的常規類來實現。
-
連接點(Joint point):是指在程序執行期間的一個點,比如某個方法的執行或者是某個異常的處理。在Spring AOP中,一個連接點往往代表的是一個方法執行。
-
通知(Advice):是指切面在某個特殊連接點上執行的動作。通知有不同類型,包括"around","before"和"after"通知。許多AOP框架包括Spring,將通知建模成一個攔截器,並且圍繞連接點維持一個攔截器鏈。
-
切入點(Pointcut):是指匹配連接點的一個斷言。通知是和一個切入點表達式關聯的,並且在任何被切入點匹配的連接點上運行(舉例,使用特定的名字執行某個方法)。AOP的核心就是切入點表達式匹配連接點的思想。Spring默認使用AspectJ切入點表達式語
-
引入(Introduction):代表了對一個類型額外的方法或者屬性的聲明。Spring AOP允許引入新接口到任何被通知對象(以及一個對應實現)。比如,可以使用一個引入去使一個bean實現IsModified接口,從而簡化緩存機制。(在AspectJ社區中,一個引入也稱為一個inter-type declaration類型間聲明)
-
目標對象(Target object):是指被一個或多個切面通知的那個對象。也指被通知對象("advised object"),由於Spring AOP是通過運行時代理事項的,這個目標對象往往是一個代理對象。
-
AOP 代理(AOP proxy):是指通過AOP框架創建的對象,用來實現切面合約的(執行通知方法等等)。在Spring框架中,一個AOP代理是一個JDK動態代理或者是一個CGLIB代理。
-
織入(Weaving):將切面和其他應用類型或者對象連接起來,創騙一個被通知對象。這些可以在編譯時(如使用AspectJ編譯器)、加載時或者運行時完成。Spring AOP,比如其他純Java AOP框架一般是在運行時完成織入。
實現
1 引用依賴包
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
2 添加切面和攔截的行為
@Aspect
@Component
@Slf4j
public class TestAspect {
/**
* 對TestService類下面的所有方法攔截.
*/
@Pointcut("execution(* com.lind.start.test.aop.TestService.*(..))")
public void pointcut() {
}
//前置通知
@Before("pointcut()")
public void beforeMethod(JoinPoint joinPoint) {
if (joinPoint.getArgs().length == 1 && joinPoint.getArgs()[0] instanceof User) {
User user = (User) joinPoint.getArgs()[0];
user.setUsername("aop賦值");
log.info("調用了前置通知" + user.toString());
}
}
//@After: 后置通知
@After("pointcut()")
public void afterMethod(JoinPoint joinPoint) {
log.info("調用了后置通知");
}
//@AfterRunning: 返回通知 result為返回內容
@AfterReturning(value = "pointcut()", returning = "result")
public void afterReturningMethod(JoinPoint joinPoint, Object result) {
log.info("調用了返回通知");
}
//@Around:環繞通知
@Around("pointcut()")
public Object Around(ProceedingJoinPoint pjp) throws Throwable {
log.info("around執行方法之前");
Object object = pjp.proceed();
log.info("around執行方法之后--返回值:" + object);
return object;
}
}
3 調用及結果
@SpringBootTest
@RunWith(SpringRunner.class)
public class AopTest {
@Autowired
TestService testService;
@Test
public void test() {
testService.print(new User());
}
}