您好,登錄后才能下訂單哦!
這篇文章主要介紹“Spring AOP怎么使用切入點創(chuàng)建通知”,在日常操作中,相信很多人在Spring AOP怎么使用切入點創(chuàng)建通知問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Spring AOP怎么使用切入點創(chuàng)建通知”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!
通過之前的例子中,我們可以創(chuàng)建ProxyFactory的方式來創(chuàng)建通知,然后獲取目標(biāo)類中的方法。通過不同類型的通知,能對這些方法做不同的事。但是,這種方式會對整個類中的所有方法都有作用,但是很多時間我們只想對這個類中的部分方法進行通知處理,那就要使用切入點來精確地控制到特定的方法
也就是說,我們的切入點就是用來確定一個類中的方法(精確到方法),類似于定義一些規(guī)則一樣,來找到和這個規(guī)則相匹配的類,知道這一點,往下看就容易多了。
在Spring中的要創(chuàng)建切入點時,就要實現(xiàn)Pointcut類。
package org.springframework.aop; public interface Pointcut{ ClassFilter getClassFilter(); MethodMatcher getMethodMacher(); }
以上兩個方法返回的類的源碼如下:
ClassFilter
package org.springframework.aop; /** * 這是一個函數(shù)式接口,就是傳入一個類, * 如果這個類滿足我們的要求,就返回true * 也就是說這個切入點適用于這個類(也就是這個類不匹配我們的規(guī)則) */ @FunctionalInterface public interface ClassFilter { boolean matches(Class<?> var1); }
MethodMatcher
package org.springframework.aop; import java.lang.reflect.Method; /** * 這個當(dāng)然就是用來匹配方法了, * 有兩種類型,動態(tài)和靜態(tài),這個由isRuntime()的返回值來決定,true就是動態(tài),false就是靜態(tài)。這個類型就是決定了這個切入點是動態(tài)的還是靜態(tài)的 * */ public interface MethodMatcher { MethodMatcher TRUE = TrueMethodMatcher.INSTANCE; //用于靜態(tài)匹配,就是和方法的參數(shù)無關(guān) boolean matches(Method var1, Class<?> var2); boolean isRuntime(); //用于動態(tài)匹配,也就是和方法的參數(shù)有關(guān),因為參數(shù)是會變的 boolean matches(Method var1, Class<?> var2, Object... var3); }
綜上,切入點分為兩種,動態(tài)切入點和靜態(tài)切入點,動態(tài)切入點要每次檢查方法的實參是不是滿足要求,這會產(chǎn)生額外的開支。如果可以,如果可以,盡可能使用靜態(tài)切入點。
實現(xiàn)類 | 描述 |
---|---|
org.springframework.aop.support.annotation.AnnotationMatchingPointcut | 在類或方法上找特定的注解,需要JDK5以上版本 |
org.springframework.aop.aspectj.AspectJExpressionPointcut | 使用AspectJ織入器以AspectJ語法評估切入點表態(tài)式 |
org.springframework.aop.support.ComposablePointcut | 使用諸如union()和intersection()等操作組合兩個或多個切入點 |
org.springframework.aop.support.ControlFlowPointcut | 是一種特殊的切入點,它們匹配另一個方法的控制流中的所有方法,即任何作為另一個方法的結(jié)果而直接或間接調(diào)用的方法 |
org.springframework.aop.support.JdkRegexpMethodPointcut | 對方法名使用正則表達式定義切入點,要JDK4以上 |
org.springframework.aop.support.NameMatchMethodPointcut | 顧名思義,這是對方法名稱列表進行簡單的匹配 |
org.springframework.aop.support.DynamicMethodMatcherPointcut | 這個類作為創(chuàng)建動態(tài)切入點的基類 |
org.springframework.aop.support.StaticMethodMatcherPointcut | 作為創(chuàng)建表態(tài)切入點的基類 |
創(chuàng)建一個類,兩個方法。我們的目的就是只在walk()方法中創(chuàng)建環(huán)繞通知,打印一句,"I am a cute cat."
public class Cat { public void sleep(){ System.out.println("sleep...."); } public void walk(){ System.out.println("walking...."); } }
創(chuàng)建切入點
public class MethodPointcutDemo extends StaticMethodMatcherPointcut { @Override public boolean matches(Method method, Class<?> aClass) { return method.getName().equals("walk"); } @Override public ClassFilter getClassFilter() { return clz -> clz == Cat.class; // 上邊的lambda表達式等于下邊的這個 // return new ClassFilter() { // @Override // public boolean matches(Class<?> clz) { // return clz == Cat.class; // } // }; } }
在上邊這個方法中,當(dāng)然,我們可以不用實現(xiàn)getClassFilter()方法,因為這個方法已經(jīng)被上級實現(xiàn)過了,我們就可以在matches方法中直接去判斷這個類是不是Cat.class
通知類(這個和上一次是一樣的,創(chuàng)建一個環(huán)繞通知)
import org.aopalliance.intercept.MethodInterceptor; import org.aopalliance.intercept.MethodInvocation; public class CatAdvisor implements MethodInterceptor{ @Override public Object invoke(MethodInvocation invocation) throws Throwable { //最不靠譜的方法,我們可以在這里判斷這個method的是不是walk,從而要不要進行通知 System.out.println("I am a cute Cat."); Object proceed = invocation.proceed(); return proceed; } }
當(dāng)然,我們在這里也是可以判斷這個方法名和類名的,為什么還要用切入點呢。可是這并不靠譜,我們中需要在這里實現(xiàn)我們的邏輯代碼,而通過切入點來控制哪個類,哪個方法要被通知,這樣更靈活。
測試方法
public static void main(String[] args) { Cat cat = new Cat(); Pointcut pointcut = new MethodPointcutDemo();//切入點實例 Advice advice = new CatAdvisor();//通知類實例(就是我們的通知代碼存放的類的對象) Advisor advisor = new DefaultPointcutAdvisor(pointcut, advice);//切面類,就是切入點和通知類的集合 ProxyFactory proxyFactory = new ProxyFactory(); //和之前代碼的區(qū)別就是這一句,這里我們使用的是切入點控制,如果把這句注釋了,就要用設(shè)置通知類 proxyFactory.addAdvisor(advisor);//設(shè)置切面類,包含切入點(控制通知點)和通知類(邏輯代碼) //如果注釋了上面一句,用這一句的話,就會對這個類中的所有方法都通知 // proxyFactory.addAdvice(advice);//設(shè)置通知類 proxyFactory.setTarget(cat); Cat proxy = (Cat) proxyFactory.getProxy(); proxy.sleep(); proxy.walk(); }
運行結(jié)果肯定就是我們想的那樣
sleep.... I am a cute Cat. walking....
這個和上邊的靜態(tài)切入點是一樣的,只不過是讓傳入方法的參數(shù)滿足一定要求時,才會執(zhí)行通知。由于篇幅原因,就不寫了,在本文最后下載代碼就能看懂。
目標(biāo)類和通知類還是之前的Cat類,之前的切入點的實現(xiàn)類不用寫,因為這個類已經(jīng)做了默認實現(xiàn),感興趣的可以看下它的源碼,很簡單的,就是匹配類名,和我們剛才創(chuàng)建的靜態(tài)切入點差不多。
public class NameMatchPointcutDemo { public static void main(String[] args) { Cat cat = new Cat(); //這個類已經(jīng)是個實現(xiàn)類,我們就不需要再去寫實現(xiàn)類了 NameMatchMethodPointcut pointcut = new NameMatchMethodPointcut(); pointcut.addMethodName("walk"); Advisor advisor = new DefaultPointcutAdvisor(pointcut,new CatAdvisor()); ProxyFactory proxyFactory = new ProxyFactory(); proxyFactory.addAdvisor(advisor); proxyFactory.setTarget(cat); Cat proxy = (Cat) proxyFactory.getProxy(); proxy.sleep(); proxy.walk(); } }
只寫測試類,其他的都和上邊一樣
public class JdkRegexpPointcutDemo { public static void main(String[] args) { Cat cat = new Cat(); JdkRegexpMethodPointcut pointcut = new JdkRegexpMethodPointcut(); pointcut.setPattern(".*ee.*");//匹配中間有ee字母的,sleep() Advisor advisor = new DefaultPointcutAdvisor(pointcut,new CatAdvisor()); ProxyFactory proxyFactory = new ProxyFactory(); proxyFactory.addAdvisor(advisor); proxyFactory.setTarget(cat); Cat proxy = (Cat) proxyFactory.getProxy(); proxy.sleep(); proxy.walk(); } }
使用AspectJ時要加入相關(guān)依賴
<dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjrt</artifactId> <version>1.9.1</version> </dependency> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.9.1</version> </dependency>
public class AspectJExpressionPointcutDemo { public static void main(String[] args) { Cat cat = new Cat(); AspectJExpressionPointcut pointcut = new AspectJExpressionPointcut(); pointcut.setExpression("execution(* walk*(..))"); Advisor advisor = new DefaultPointcutAdvisor(pointcut, new CatAdvisor()); ProxyFactory proxyFactory = new ProxyFactory(); proxyFactory.addAdvisor(advisor); proxyFactory.setTarget(cat); Cat proxy = (Cat) proxyFactory.getProxy(); proxy.sleep(); proxy.walk(); } }
這個execution表達式的意思是:任何以walk開頭的,具有任何參數(shù)和任何返回值的方法
首先自定義一個注解,如果不是很懂,參考Java中自定義注解類,并加以運用
/** * 這個注解是用于運行時期、可用于類、方法上 */ @Retention(RetentionPolicy.RUNTIME) @Target({ElementType.TYPE,ElementType.METHOD}) public @interface MyAdvice { }
然后在目標(biāo)方法上添加這個注解(要被通知的類)
/** * 為了不與之前的沖突,就新寫了一個方法 */ @MyAdvice public void eat(){ System.out.println("eating...."); }
然后在main方法中指定這個注解名:
public class AnnotationPointcutDemo { public static void main(String[] args) { Cat cat = new Cat(); AnnotationMatchingPointcut pointcut = AnnotationMatchingPointcut .forMethodAnnotation(MyAdvice.class); //這個類還有一個.forClassAnnotation()方法,就是指定類的 Advisor advisor = new DefaultPointcutAdvisor(pointcut, new CatAdvisor()); ProxyFactory proxyFactory = new ProxyFactory(); proxyFactory.addAdvisor(advisor); proxyFactory.setTarget(cat); Cat proxy = (Cat) proxyFactory.getProxy(); proxy.sleep();//不會被通知 proxy.walk();//不會被通知 proxy.eat();//會被通知 } }
到此,關(guān)于“Spring AOP怎么使用切入點創(chuàng)建通知”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注億速云網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。