非Spring管理Bean如何添加AOP呢?
前幾天有個朋友問了一個問題,覺得可以給大家分享一下。
問題如下圖
歸其根本這是個歷史項目,里面有很多的類并沒有交給spring管理,但現在需要統一添加日志。
面對這樣的問題,其實只要了解AOP的原理,就會有多種方法。AOP都是基于動態代理來實現,而動態代理常見的就是cglib和java動態代理,不了解的可以看下之前干貨君寫的文章
- java動態代理為什么需要基于接口
- cglib動態代理對類沒有任何限制嗎?
但此兩種方法似乎在這樣的場景不好實現,需要修改大量的代碼,那么有沒有什么好的方案呢?
答案當然是有。
首先要清楚的是AOP的底層實現原理就是字節碼,我們只需要從字節碼層面,就一定可以解決這樣的問題。因此可以利用編譯期增強和運行期增強,常見的方案有兩種,一種Java Agent技術,另一種 AspectJ方案。
Java Agent
Java Agent中文名字叫做java 探針,可以在運行java時指定探針程序,對原程序無侵入,常見的一些APM工具都會這樣,如skywalking,后續有機會給大家介紹下。如下圖

java agent的主要原理就是利用JVMTI(JVM Tool Interface),JVM用來暴露一些供用戶擴展的接口集合,因此可以在此處做一些運行期字節碼增強。
Java Agent內容比較多,有很多大家熟悉的工具都是基于它去做的,例如阿里的arthas。本文就不介紹了,后期會給大家詳細介紹下Java Agent。
AspectJ方案
可以利用aspectj + javac來編譯織入代碼,也可以利用maven插件aspectj-maven-plugin,下面利用AspectJ注解 + aspectj-maven-plugin來實戰一下。
aspectj-maven-plugin官網 http://www.mojohaus.org/aspectj-maven-plugin/usage.html
引入依賴
編譯增強,依賴此jar
- import org.aspectj.lang.annotation.Aspect;
- import org.aspectj.lang.annotation.Before;
- import org.aspectj.lang.annotation.Pointcut;
- @Aspect
- public class Aop {
- @Pointcut("execution(* com.ganhuojun.gracefulshutdown.controller..*.*(..))")
- public void pointcut1(){
- }
- @Before("pointcut1()")
- public void before(){
- System.out.println("controller before");
- }
- }
定義注解
注意:該注解不要交給spring管理
- <plugin>
- <groupId>org.codehaus.mojo</groupId>
- <artifactId>aspectj-maven-plugin</artifactId>
- <version>1.11</version>
- <configuration>
- <complianceLevel>1.8</complianceLevel>
- <source>1.8</source>
- <!--<showWeaveInfo>true</showWeaveInfo>-->
- <!--<Xlint>ignore</Xlint>-->
- <encoding>UTF-8</encoding>
- <sources>
- <source>
- <basedir>src/main/java</basedir>
- <!--此處使用include一致會導致織入失敗,暫時未找到好的解決辦法,不寫則引用所有的Aspect-->
- <!--<includes>-->
- <!--<include>**/Aop.java</include>-->
- <!--<include>**/ControllerAop.aj</include>-->
- <!--</includes>-->
- <excludes>
- <exclude>**/ServiceAop.java</exclude>
- </excludes>
- </source>
- </sources>
- </configuration>
- <executions>
- <execution>
- <goals>
- <goal>compile</goal>
- </goals>
- </execution>
- </executions>
- </plugin>
配置maven插件
- <plugin>
- <groupId>org.codehaus.mojo</groupId>
- <artifactId>aspectj-maven-plugin</artifactId>
- <version>1.11</version>
- <configuration>
- <complianceLevel>1.8</complianceLevel>
- <source>1.8</source>
- <!--<showWeaveInfo>true</showWeaveInfo>-->
- <!--<Xlint>ignore</Xlint>-->
- <encoding>UTF-8</encoding>
- <sources>
- <source>
- <basedir>src/main/java</basedir>
- <!--此處使用include一致會導致織入失敗,暫時未找到好的解決辦法,不寫則引用所有的Aspect-->
- <!--<includes>-->
- <!--<include>**/Aop.java</include>-->
- <!--<include>**/ControllerAop.aj</include>-->
- <!--</includes>-->
- <excludes>
- <exclude>**/ServiceAop.java</exclude>
- </excludes>
- </source>
- </sources>
- </configuration>
- <executions>
- <execution>
- <goals>
- <goal>compile</goal>
- </goals>
- </execution>
- </executions>
- </plugin>
排除spring的aop
如果對spring aop比較熟悉的都知道,spring的aop也是基于AspectJ的,因此需要exclude的,已經配置到mavn的地方了。
編譯&運行&測試
編譯后class文件已經被織入了相關代碼,如下圖

運行相關日志輸出如下

說明功能已經實現。