溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Spring AOP怎么使用切入點創(chuàng)建通知

發(fā)布時間:2022-03-14 16:03:31 來源:億速云 閱讀:164 作者:iii 欄目:web開發(fā)

這篇文章主要介紹“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)類

實現(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)切入點的基類

四、使用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....

五、使用DyanmicMatcherPointcut創(chuàng)建動態(tài)切入點

這個和上邊的靜態(tài)切入點是一樣的,只不過是讓傳入方法的參數(shù)滿足一定要求時,才會執(zhí)行通知。由于篇幅原因,就不寫了,在本文最后下載代碼就能看懂。

六、其他類型的PointCut的實現(xiàn)類

1.簡單的名稱匹配 ( NameMatchMethodPointcut )

目標(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();
    }
}
2.使用正則表達式創(chuàng)建切入點 ( JdkRegexpMethodPointcut )

只寫測試類,其他的都和上邊一樣

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();
    }
}
3.使用AspectJ切入點表達式創(chuàng)建切入點 ( AspectJExpressionPointcut )

使用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ù)和任何返回值的方法

4.創(chuàng)建注解匹配的切入點 ( AnnotationMatchingPointcut )

首先自定義一個注解,如果不是很懂,參考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>

向AI問一下細節(jié)

免責(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)容。

AI