一、AOP是什麼?
AOP的全稱是 Aspect Oriented Program(面向切面程式設計)
按照我的理解,AOP的思想通俗易懂的來說就是將程式的功能分為兩類,一類是核心功能,另一類是輔助功能。
在非面向切面程式設計裡,核心功能和輔助功能總是耦合在一起,比如說進行資料庫操作時,日誌記錄行為的程式碼總是緊挨著資料庫操作(不知道別的大神是什麼樣,反正我是這樣)。這樣一來,核心功能(資料庫操作)的程式碼就和輔助功能(記錄日誌)的程式碼耦合在一起了。
而在AOP裡,核心功能和輔助功能的程式碼是完全分開的,它們可以各自獨立修改,而完全不干擾對方,這實際上應該是應用了Java設計模式裡的代理模式(改天記得複習一下)。
二、Spring如何應用AOP?
1.準備工作有哪些
-
要加哪些包?
spring的核心包就不用說了,除此之外還要加上和AOP有關的包,aspectJ,完整的maven配置檔案如下:
<?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>SpringLearn</groupId> <artifactId>SpringDemo</artifactId> <version>1.0-SNAPSHOT</version> <dependencies> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aop</artifactId> <version>4.3.11.RELEASE</version> </dependency> <dependency> <groupId>aopalliance</groupId> <artifactId>aopalliance</artifactId> <version>1.0</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-beans</artifactId> <version>4.3.11.RELEASE</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-core</artifactId> <version>4.3.11.RELEASE</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>4.3.11.RELEASE</version> </dependency> <!-- https://mvnrepository.com/artifact/org.aspectj/aspectjrt --> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjrt</artifactId> <version>1.9.1</version> </dependency> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.6.8</version> </dependency> <dependency> <groupId>commons-logging</groupId> <artifactId>commons-logging</artifactId> <version>1.2</version> </dependency> </dependencies> <build> <plugins> <plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-eclipse-plugin</artifactId> <version>2.8</version> <configuration> <downloadSources>true</downloadSources> </configuration> </plugin> </plugins> </build> </project> 複製程式碼
-
要加哪些類?
LoggerAspect
ProductService
Category
Product
目錄結構如下:
完整程式碼在這2.核心功能 和 輔助功能 是怎麼分開的?
建立一個核心功能的類 ProductService
package com.learn.service;
import org.springframework.stereotype.Component;
@Component("ps")
public class ProductService {
public void doSomeService() {
System.out.println("假裝這裡是一些核心業務程式碼!");
}
}
複製程式碼
建立一個輔助功能的類 LoggerAspect
package com.learn.aspect;
import org.springframework.stereotype.Component;
import org.aspectj.lang.ProceedingJoinPoint;
@Component("la")
public class LoggerAspect {
public Object log(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("start log:" + joinPoint.getSignature().getName());
Object object = joinPoint.proceed();
System.out.println("end log:" + joinPoint.getSignature().getName());
return object;
}
}
複製程式碼
可以看到這裡核心功能的程式碼和輔助功能的程式碼是完全分開的,那麼怎麼把它們聯絡到一起呢?
這就需要配置Spring的配置檔案了,具體配置如下:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
<!--掃描bean-->
<context:component-scan base-package="com.learn.service com.learn.springBasic com.learn.aspect"/>
<!--配置AOP-->
<aop:config>
<!--配置切點-->
<aop:pointcut id="loggerCutpoint" expression="execution(* com.learn.service.ProductService.*(..)) "/>
<!--配置織入物件-->
<aop:aspect id="logAspect" ref="la">
<aop:around pointcut-ref="loggerCutpoint" method="log"/>
</aop:aspect>
</aop:config>
</beans>
複製程式碼
配置檔案中需要加入 <aop:config> 標籤。
<aop:pointcut> 標籤配置的是核心類,在這裡指定了要織入的物件是哪個。
execution(* com.learn.service.ProductService.*(..))
複製程式碼
上面這句表示式表示,所有ProductService下的方法都將被織入輔助功能,既在執行該方法時會呼叫輔助功能。 <aop:aspect> 配置的是輔助類
<aop:around pointcut-ref="loggerCutpoint" method="log"/>
複製程式碼
上面標籤中的 pointcut-ref 很顯然就是要織入的核心類id, method 就是被織入的方法。
執行結果:
start log:doSomeService
假裝這裡是一些核心業務程式碼!
end log:doSomeService
複製程式碼
執行結果呈現出這樣的效果的是因為輔助類中log方法中是這樣寫的:
System.out.println("start log:" + joinPoint.getSignature().getName());
Object object = joinPoint.proceed();
System.out.println("end log:" + joinPoint.getSignature().getName());
複製程式碼
log方法最終返回的是一個Object物件,就本例來說,返回的實際上是ProductService 物件,這裡應該是用到了代理模式來實現。
三、用註解方式配置AOP
註解方式配置AOP也很簡單,和註解方式配置bean是非常相似的。
首先,spring配置檔案中把 <aop:config> 下的標籤都刪掉,換成 <aop:aspectj-autoproxy/> 標籤來啟用自動代理。
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
<!--掃描bean-->
<context:component-scan base-package="com.learn.service com.learn.springBasic com.learn.aspect"/>
<!--配置自動代理AOP-->
<aop:aspectj-autoproxy/>
</beans>
複製程式碼
LoggerAspect 類中也要做一些改變,加上@Aspect 和 @Around註解。
package com.learn.aspect;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
import org.aspectj.lang.ProceedingJoinPoint;
@Aspect
@Component("la")
public class LoggerAspect {
@Around(value = "execution(* com.learn.service.ProductService.*(..))")
public Object log(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("start log:" + joinPoint.getSignature().getName());
Object object = joinPoint.proceed();
System.out.println("end log:" + joinPoint.getSignature().getName());
return object;
}
}
複製程式碼
執行結果:
start log:doSomeService
假裝這裡是一些核心業務程式碼!
end log:doSomeService
複製程式碼