日日操夜夜添-日日操影院-日日草夜夜操-日日干干-精品一区二区三区波多野结衣-精品一区二区三区高清免费不卡

公告:魔扣目錄網為廣大站長提供免費收錄網站服務,提交前請做好本站友鏈:【 網站目錄:http://www.ylptlb.cn 】, 免友鏈快審服務(50元/站),

點擊這里在線咨詢客服
新站提交
  • 網站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會員:747

本文內容

  1. Spring 10種切點表達式詳解
  2. 切點的組合使用
  3. 公共切點的定義

聲明切點@Poincut

@Poincut 的使用格式如下:

@Poincut("PCD") // 切點表達式 表示對哪些方法進行增強
public void pc(){} // 切點簽名,返回值必須為void

10種切點表達式

AspectJ的切點指示符AspectJ pointcut designators (PCD) ,也就是俗稱的切點表達式,Spring中支持10種,如下表:

表達式類型

作用

匹配規則

execution

用于匹配方法執行的連接點

 

within

用于匹配指定類型內的方法執行

within(x)匹配規則target.getClass().equals(x)

this

用于匹配當前AOP代理對象類型的執行方法,包含引入的接口類型匹配

this(x)匹配規則:
x.getClass.isAssingableFrom(proxy.getClass)

target

用于匹配當前目標對象類型的執行方法,不包括引入接口的類型匹配

target(x)匹配規則:x.getClass().isAssignableFrom(target.getClass());

args

用于匹配當前執行的方法傳入的參數為指定類型的執行方法

傳入的目標位置參數.getClass().equals(@args(對應的參數位置的注解類型))!= null

@target

用于匹配當前目標對象類型的執行方法,其中目標對象持有指定的注解


target.class.getAnnotation(指定的注解類型) != null

@args

用于匹配當前執行的方法傳入的參數持有指定注解的執行

傳入的目標位置參數.getClass().getAnnotation(@args(對應的參數位置的注解類型))!= null

@within

用于匹配所有持有指定注解類型內的方法

被調用的目標方法Method對象.getDeclaringClass().getAnnotation(within中指定的注解類型) != null

@annotation

用于匹配當前執行方法持有指定注解的方法

target.getClass().getMethod("目標方法名").getDeclaredAnnotation(@annotation(目標注解))!=null

bean

Spring AOP擴展的,AspectJ沒有對應的指示符,用于匹配特定名稱的Bean對象的執行方法


ApplicationContext.getBean("bean表達式中指定的bean名稱") != null

簡單介紹下AspectJ中常用的3個通配符:

  • *:匹配任何數量的字符
  • ..:匹配任何數量字符的重復,如任何數量子包,任何數量方法參數
  • +:匹配指定類型及其子類型,僅作為后綴防過載類型模式后面。

execution

用于匹配方法執行,最常用。

格式說明

   execution(modifiers-pattern? ret-type-pattern declaring-type-pattern?name-pattern(param-pattern)
                throws-pattern?)
  • 其中帶 ?號的 modifiers-pattern?,declaring-type-pattern?,throws-pattern?是可選項
  • ret-type-pattern,name-pattern, parameters-pattern是必選項
  • modifier-pattern? 修飾符匹配,如public 表示匹配公有方法,*表示任意修飾符
  • ret-type-pattern 返回值匹配,* 表示任何返回值,全路徑的類名等
  • declaring-type-pattern? 類路徑匹配
  • name-pattern 方法名匹配,* 代表所有,xx*代表以xx開頭的所有方法
  • (param-pattern) 參數匹配,指定方法參數(聲明的類型),(..)代表所有參數,(*,String)代表第一個參數為任何值,第二個為String類型,(..,String)代表最后一個參數是String類型
  • throws-pattern? 異常類型匹配

舉例說明

public class PointcutExecution {

    // com.crab.spring.aop.demo02包下任何類的任意方法
    @Pointcut("execution(* com.crab.spring.aop.demo02.*.*(..))")
    public void m1(){}

    // com.crab.spring.aop.demo02包及其子包下任何類的任意方法
    @Pointcut("execution(* com.crab.spring.aop.demo02..*.*(..))")
    public void m2(){}

    // com.crab.spring.aop包及其子包下IService接口的任意無參方法
    @Pointcut("execution(* com.crab.spring.aop..IService.*(..))")
    public void m3(){}

    // com.crab.spring.aop包及其子包下IService接口及其子類型的任意無參方法
    @Pointcut("execution(* com.crab.spring.aop..IService+.*(..))")
    public void m4(){}

    // com.crab.spring.aop.demo02.UserService類中有且只有一個String參數的方法
    @Pointcut("execution(* com.crab.spring.aop.demo02.UserService.*(String))")
    public void m5(){}

    // com.crab.spring.aop.demo02.UserService類中參數個數為2且最后一個參數類型是String的方法
    @Pointcut("execution(* com.crab.spring.aop.demo02.UserService.*(*,String))")
    public void m6(){}

    // com.crab.spring.aop.demo02.UserService類中最后一個參數類型是String的方法
    @Pointcut("execution(* com.crab.spring.aop.demo02.UserService.*(..,String))")
    public void m7(){}
}

within

格式說明

within(類型表達式):目標對象target的類型是否和within中指定的類型匹配

匹配規則: target.getClass().equals(within表達式中指定的類型)

舉例說明

public class PointcutWithin {
    // 匹配 com.crab.spring.aop.demo02包及其子包下任何類的任何方法
    @Pointcut("within(com.crab.spring.aop.demo02..*)")
    public void m() {
    }

    // 匹配m.crab.spring.aop.demo02包及其子包下IService類型及其子類型的任何方法
    @Pointcut("within(com.crab.spring.aop.demo02..IService+)")
    public void m2() {
    }

    // 匹配com.crab.spring.aop.demo02.UserService類中所有方法,不含其子類
    @Pointcut("within(com.crab.spring.aop.demo02.UserService)")
    public void m3() {
    }
}

this

格式說明

this(類型全限定名):通過aop創建的代理對象的類型是否和this中指定的類型匹配;this中使用的表達式必須是類型全限定名,不支持通配符。

this(x)的匹配規則是:x.getClass.isAssingableFrom(proxy.getClass)

舉例說明

package com.crab.spring.aop.demo02.aspectj;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.aop.aspectj.annotation.AspectJProxyFactory;
import org.springframework.aop.support.AopUtils;
import org.springframework.util.ClassUtils;

/**
 * @author zfd
 * @version v1.0
 * @date 2022/2/6 21:41
 */
@Aspect
public class PointcutThis {
    interface I1{
        void m();
    }
    static class C1 implements I1{

        @Override
        public void m() {
            System.out.println("C1 m()");
        }
    }
	// 匹配 I1類型或是其子類
    @Pointcut("this(com.crab.spring.aop.demo02.aspectj.PointcutThis.I1)")
    public void pc(){}

    @Before("pc()")
    public void before(JoinPoint joinPoint) {
        System.out.println("before: " + joinPoint);
    }

    public static void main(String[] args) {
        C1 target = new C1();
        AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
        proxyFactory.setTarget(target);
        // proxyFactory.setProxyTargetClass(true);
        // 獲取C1上所有接口 spring工具類提供的方法
        Class<?>[] allInterfaces = ClassUtils.getAllInterfaces(target);
        // 設置代理接口
        proxyFactory.setInterfaces(allInterfaces);
        // 添加切面
        proxyFactory.addAspect(PointcutThis.class);
        // 獲取代理
        I1 proxy = proxyFactory.getProxy();
        // 調用方法
        proxy.m();
        System.out.println("JDK代理? " + AopUtils.isJdkDynamicProxy(proxy));
        System.out.println("CGLIB代理? " + AopUtils.isCglibProxy(proxy));
        //判斷代理對象是否是C1類型的
        System.out.println(C1.class.isAssignableFrom(proxy.getClass()));
    }

}

來觀察下輸出

before: execution(void com.crab.spring.aop.demo02.aspectj.PointcutThis$C1.m())
C1 m()
JDK代理? false
CGLIB代理? true
true

使用JDK動態代理生成的代理對象,其類型是I1類型。

思考下:將切點表達式改成下面的輸出結果是?

// 匹配 C1類型或是其子類
@Pointcut("this(
com.crab.spring.aop.demo02.aspectj.PointcutThis.C1)")
public void pc(){}

target

格式說明

target(類型全限定名):判斷目標對象的類型是否和指定的類型匹配;表達式必須是類型全限定名,不支持通配符。

target(x)匹配規則:x.getClass().isAssignableFrom(target.getClass());

舉例說明

@Aspect
public class PointcutTarget {
    interface I1{
        void m();
    }
    static class C1 implements I1{

        @Override
        public void m() {
            System.out.println("C1 m()");
        }
    }

    // 匹配目標類型必須是
    @Pointcut("target(com.crab.spring.aop.demo02.aspectj.PointcutTarget.C1)")
    public void pc(){}

    @Before("pc()")
    public void before(JoinPoint joinPoint) {
        System.out.println("before: " + joinPoint);
    }

    public static void main(String[] args) {
        C1 target = new C1();
        AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
        proxyFactory.setTarget(target);
        proxyFactory.setProxyTargetClass(true);
        // 獲取C1上所有接口 spring工具類提供的方法
        Class<?>[] allInterfaces = ClassUtils.getAllInterfaces(target);
        // 設置代理接口
        proxyFactory.setInterfaces(allInterfaces);
        // 添加切面
        proxyFactory.addAspect(PointcutTarget.class);
        // 獲取代理
        I1 proxy = proxyFactory.getProxy();
        // 調用方法
        proxy.m();
        System.out.println("JDK代理? " + AopUtils.isJdkDynamicProxy(proxy));
        System.out.println("CGLIB代理? " + AopUtils.isCglibProxy(proxy));
        //判斷代理對象是否是C1類型的
        System.out.println(C1.class.isAssignableFrom(target.getClass()));
    }

}

輸出結果

before: execution(void com.crab.spring.aop.demo02.aspectj.PointcutTarget$C1.m())
C1 m()
JDK代理? false
CGLIB代理? true
true

args

格式說明

args(參數類型列表)匹配當前執行的方法傳入的參數是否為args中指定的類型;參數類型列表中的參數必須是類型全限定名,不支持通配符;args屬于動態切入點,也就是執行方法的時候進行判斷的,開銷非常大,非特殊情況最好不要使用。

args(String) //    方法個數為1,類型是String
args(*,String) //  方法參數個數2,第2個是String類型
args(..,String) // 方法個數不限制,最后一個必須是String

舉例說明

package com.crab.spring.aop.demo02.aspectj;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.aop.aspectj.annotation.AspectJProxyFactory;
import org.springframework.aop.support.AopUtils;
import org.springframework.util.ClassUtils;

/**
 * @author zfd
 * @version v1.0
 * @date 2022/2/6 21:41
 */
@Aspect
public class PointcutArgs {
    interface I1{
        void m(Object name);
    }
    static class C1 implements I1{

        @Override
        public void m(Object name) {
            String type = name.getClass().getName();
            System.out.println("C1 m() 參數類型 " + type);
        }
    }

    // 匹配方法參數個數1且類型是必須是String
    @Pointcut("args(String)")
    public void pc(){}

    @Before("pc()")
    public void before(JoinPoint joinPoint) {
        System.out.println("before: " + joinPoint);
    }

    public static void main(String[] args) {
        C1 target = new C1();
        AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
        proxyFactory.setTarget(target);
        proxyFactory.setProxyTargetClass(true);
        // 獲取C1上所有接口 spring工具類提供的方法
        Class<?>[] allInterfaces = ClassUtils.getAllInterfaces(target);
        // 設置代理接口
        proxyFactory.setInterfaces(allInterfaces);
        // 添加切面
        proxyFactory.addAspect(PointcutArgs.class);
        // 獲取代理
        I1 proxy = proxyFactory.getProxy();
        // 調用方法
        proxy.m("xxxx");
        proxy.m(100L);
        System.out.println("JDK代理? " + AopUtils.isJdkDynamicProxy(proxy));
        System.out.println("CGLIB代理? " + AopUtils.isCglibProxy(proxy));
        //判斷代理對象是否是C1類型的
        System.out.println(C1.class.isAssignableFrom(target.getClass()));
    }

}

觀察下輸出

before: execution(void com.crab.spring.aop.demo02.aspectj.PointcutArgs$C1.m(Object))
C1 m() 參數類型 JAVA.lang.String
C1 m() 參數類型 java.lang.Long
JDK代理? false
CGLIB代理? true
true	

參數類型傳遞是String時候增強了,而Long的時候沒有執行增強方法。

@within

格式說明

@within(注解類型):匹配指定的注解內定義的方法。

匹配規則: 被調用的目標方法Method對象.getDeclaringClass().getAnnotation(within中指定的注解類型) != null

舉例說明

package com.crab.spring.aop.demo02.aspectj;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.aop.aspectj.annotation.AspectJProxyFactory;
import org.springframework.aop.support.AopUtils;
import org.springframework.util.ClassUtils;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * @author zfd
 * @version v1.0
 * @date 2022/2/6 21:41
 * @關于我 請關注公眾號 螃蟹的Java筆記 獲取更多技術系列
 */
@Aspect
public class PointcutAnnWithin {
    @Retention(RetentionPolicy.RUNTIME)
    @Target(ElementType.TYPE)
    @interface MyAnn {
    }

    interface I1 {
        void m();
    }

    @MyAnn
    static class C1 implements I1 {
        @Override
        public void m() {
            System.out.println("C1 m()");
        }
    }

    // 匹配目標類型必須上必須有注解MyAnn
    @Pointcut("@within(com.crab.spring.aop.demo02.aspectj.PointcutAnnWithin.MyAnn)")
    public void pc() {
    }

    @Before("pc()")
    public void before(JoinPoint joinPoint) {
        System.out.println("before: " + joinPoint);
    }

    public static void main(String[] args) {
        C1 target = new C1();
        AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
        proxyFactory.setTarget(target);
        proxyFactory.setProxyTargetClass(true);
        // 獲取C1上所有接口 spring工具類提供的方法
        Class<?>[] allInterfaces = ClassUtils.getAllInterfaces(target);
        // 設置代理接口
        proxyFactory.setInterfaces(allInterfaces);
        // 添加切面
        proxyFactory.addAspect(PointcutAnnWithin.class);
        // 獲取代理
        I1 proxy = proxyFactory.getProxy();
        // 調用方法
        proxy.m();
        System.out.println("JDK代理? " + AopUtils.isJdkDynamicProxy(proxy));
        System.out.println("CGLIB代理? " + AopUtils.isCglibProxy(proxy));
        //判斷代理對象是否是C1類型的
        System.out.println(C1.class.isAssignableFrom(target.getClass()));
    }

}

輸出

before: execution(void com.crab.spring.aop.demo02.aspectj.PointcutAnnWithin$C1.m())
C1 m()
JDK代理? false
CGLIB代理? true
true

思考下父類上有注解,子類繼承父類的方法,同時考慮下注解@Inherited是否在切點注解的場景?

@target

格式說明

@target(注解類型):判斷目標對象target類型上是否有指定的注解;@target中注解類型也必須是全限定類型名。

匹配規則: target.class.getAnnotation(指定的注解類型) != null

注意,如果目標注解是標注在父類上的,那么定義目標注解時候應使用@Inherited標注,使子類能繼承父類的注解。

舉例說明

package com.crab.spring.aop.demo02.aspectj;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.aop.aspectj.annotation.AspectJProxyFactory;
import org.springframework.aop.support.AopUtils;
import org.springframework.util.ClassUtils;

import java.lang.annotation.*;

/**
 * @author zfd
 * @version v1.0
 * @date 2022/2/6 21:41
 */
@Aspect
public class PointcutAnnTarget {
    @Retention(RetentionPolicy.RUNTIME)
    @Target(ElementType.TYPE)
    @Inherited // 子類能繼承父類的注解
    @interface MyAnn2 {
    }

    @MyAnn2 // 注解在父類上
    static class P1 {
        void m(){}
    }

    static class C1 extends P1 {
        @Override
        public void m() {
            System.out.println("C1 m()");
        }
    }

    // 匹配目標類型必須上必須有注解MyAnn
    @Pointcut("@target(com.crab.spring.aop.demo02.aspectj.PointcutAnnTarget.MyAnn2)")
    public void pc() {
    }

    @Before("pc()")
    public void before(JoinPoint joinPoint) {
        System.out.println("before: " + joinPoint);
    }

    public static void main(String[] args) {
        C1 target = new C1();
        AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
        proxyFactory.setTarget(target);
        proxyFactory.setProxyTargetClass(true);
        // 獲取C1上所有接口 spring工具類提供的方法
        Class<?>[] allInterfaces = ClassUtils.getAllInterfaces(target);
        // 設置代理接口
        proxyFactory.setInterfaces(allInterfaces);
        // 添加切面
        proxyFactory.addAspect(PointcutAnnTarget.class);
        // 獲取代理
        C1 proxy = proxyFactory.getProxy();
        // 調用方法
        proxy.m();
        System.out.println("JDK代理? " + AopUtils.isJdkDynamicProxy(proxy));
        System.out.println("CGLIB代理? " + AopUtils.isCglibProxy(proxy));
        // 目標類上是否有切點注解
        System.out.println(target.getClass().getAnnotation(MyAnn2.class)!= null);
    }

}

輸出結果

before: execution(void com.crab.spring.aop.demo02.aspectj.PointcutAnnTarget$C1.m())
C1 m()
JDK代理? false
CGLIB代理? true
true

從結果最后一行看,目標對象繼承了父類的注解,符合@target的切點規則。

@args

格式說明

@args(注解類型):方法參數所屬的類上有指定的注解;注意不是參數上有指定的注解,而是參數類型的類上有指定的注解。和args類似,不過針對的是參數類型上的注解。

匹配規則: 傳入的目標位置參數.getClass().getAnnotation(@args(對應的參數位置的注解類型))!= null

舉例說明

package com.crab.spring.aop.demo02.aspectj;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.aop.aspectj.annotation.AspectJProxyFactory;
import org.springframework.aop.support.AopUtils;
import org.springframework.util.ClassUtils;

import java.lang.annotation.*;

/**
 * @author zfd
 * @version v1.0
 * @date 2022/2/6 21:41
 */
@Aspect
public class PointcutAnnArgs {
    @Retention(RetentionPolicy.RUNTIME)
    @Target(ElementType.TYPE)
    @Inherited // 子類能繼承父類的注解
    @interface MyAnn3 {
    }

    @MyAnn3
    static class MyParameter{

    }

    static class C1  {
        public void m(MyParameter myParameter) {
            System.out.println(myParameter.getClass().getAnnotation(MyAnn3.class));
            System.out.println("C1 m()");
        }
    }

    // 匹配方法上最后的一個參數類型上有注解MyAnn3
    @Pointcut("@args(..,com.crab.spring.aop.demo02.aspectj.PointcutAnnArgs.MyAnn3)")
    public void pc() {
    }

    @Before("pc()")
    public void before(JoinPoint joinPoint) {
        System.out.println("before: " + joinPoint);
    }

    public static void main(String[] args) {
        C1 target = new C1();
        AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
        proxyFactory.setTarget(target);
        proxyFactory.setProxyTargetClass(true);
        // 獲取C1上所有接口 spring工具類提供的方法
        Class<?>[] allInterfaces = ClassUtils.getAllInterfaces(target);
        // 設置代理接口
        proxyFactory.setInterfaces(allInterfaces);
        // 添加切面
        proxyFactory.addAspect(PointcutAnnArgs.class);
        // 獲取代理
        C1 proxy = proxyFactory.getProxy();
        // 調用方法
        MyParameter myParameter = new MyParameter();
        proxy.m(myParameter);
        System.out.println("JDK代理? " + AopUtils.isJdkDynamicProxy(proxy));
        System.out.println("CGLIB代理? " + AopUtils.isCglibProxy(proxy));
        // 目標類上是否有切點注解
        System.out.println(myParameter.getClass().getAnnotation(MyAnn3.class)!= null);
    }

}

觀察結果

before: execution(void com.crab.spring.aop.demo02.aspectj.PointcutAnnArgs$C1.m(MyParameter))
@com.crab.spring.aop.demo02.aspectj.PointcutAnnArgs$MyAnn3()
C1 m()
JDK代理? false
CGLIB代理? true
true

第二行中目標方法上輸出了參數的注解。

最后一行判斷參數類型上確實有注解。

@annotation

格式說明

@annotation(注解類型):匹配被調用的目標對象的方法上有指定的注解

匹配規則:target.getClass().getMethod("目標方法名").getDeclaredAnnotation(@annotation(目標注解))!=null

這個在針對特定注解的方法日志攔截場景下應用比較多。

舉例說明

package com.crab.spring.aop.demo02.aspectj;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.aop.aspectj.annotation.AspectJProxyFactory;
import org.springframework.aop.support.AopUtils;
import org.springframework.util.ClassUtils;

import java.lang.annotation.*;

/**
 * @author zfd
 * @version v1.0
 * @date 2022/2/6 21:41
 */
@Aspect
public class PointcutAnnotation {
    @Retention(RetentionPolicy.RUNTIME)
    @Target(ElementType.METHOD)
    @interface MyAnn4 {
    }

    /**
     * 父類 方法上都有@MyAnn4
     */
    static class P1{
        @MyAnn4
        public void m1(){
            System.out.println("P1 m()");
        }
        @MyAnn4
        public void m2(){
            System.out.println("P1 m2()");
        }
    }

    /**
     * 子類
     * 注意重新重寫了父類的m1方法但是沒有聲明注解@Ann4
     * 新增了m3方法帶注解@Ann4
     */
    static class C1 extends P1 {
        @Override
        public void m1() {
            System.out.println("C1 m1()");
        }

        @MyAnn4
        public void m3() {
            System.out.println("C1 m3()");
        }
    }

    // 匹配調用的方法上必須有注解
    @Pointcut("@annotation(com.crab.spring.aop.demo02.aspectj.PointcutAnnotation.MyAnn4)")
    public void pc() {
    }

    @Before("pc()")
    public void before(JoinPoint joinPoint) {
        System.out.println("before: " + joinPoint);
    }

    public static void main(String[] args) throws NoSuchMethodException {
        C1 target = new C1();
        AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
        proxyFactory.setTarget(target);
        proxyFactory.setProxyTargetClass(true);
        // 獲取C1上所有接口 spring工具類提供的方法
        Class<?>[] allInterfaces = ClassUtils.getAllInterfaces(target);
        // 設置代理接口
        proxyFactory.setInterfaces(allInterfaces);
        // 添加切面
        proxyFactory.addAspect(PointcutAnnotation.class);
        // 獲取代理
        C1 proxy = proxyFactory.getProxy();
        // 調用方法
        proxy.m1();
        proxy.m2();
        proxy.m3();

        System.out.println("JDK代理? " + AopUtils.isJdkDynamicProxy(proxy));
        System.out.println("CGLIB代理? " + AopUtils.isCglibProxy(proxy));

        // 目標對象的目標方法上是否直接聲明了注解MyAnn4
        System.out.println(target.getClass().getMethod("m1").getDeclaredAnnotation(MyAnn4.class)!=null);
        System.out.println(target.getClass().getMethod("m2").getDeclaredAnnotation(MyAnn4.class)!=null);
        System.out.println(target.getClass().getMethod("m3").getDeclaredAnnotation(MyAnn4.class)!=null);
    }

}

觀察下結果

C1 m1()
before: execution(void com.crab.spring.aop.demo02.aspectj.PointcutAnnotation$P1.m2())
P1 m2()
before: execution(void com.crab.spring.aop.demo02.aspectj.PointcutAnnotation$C1.m3())
C1 m3()
JDK代理? false
CGLIB代理? true
false
true
true

簡單分析下:

  1. C1中重寫了m1方法,上面有沒有 @Ann4,所有方法沒有被攔截
  2. 其它的m2在父類上有注解@Ann4,m3在子類上也有注解@Ann4,所以攔截了。
  3. 最后3行輸出了目標對象的3個方法上是否有注解的情況。

bean

格式說明

bean(bean名稱):這個用在spring環境中,匹配容器中指定名稱的bean。

匹配格式:ApplicationContext.getBean("bean表達式中指定的bean名稱") != null

舉例說明

定義一個bean

package com.crab.spring.aop.demo02.aspectj;

/**
 * @author zfd
 * @version v1.0
 * @date 2022/2/6 23:30
 */
public class MyBean {
    private String beanName;

    public MyBean(String beanName) {
        this.beanName = beanName;
    }

    public void m() {
        System.out.println("我是" + this.beanName);
    }
}

切面中的切點和通知定義

@Aspect
public class PointcutBean {
    // 容器中bean名稱是"myBean1"的方法進行攔截
    @Pointcut("bean(myBean1)")
    public void pc() {
    }

    @Before("pc()")
    public void m(JoinPoint joinPoint) {
        System.out.println("start " + joinPoint);
    }
}

組合使用

@Aspect
@Configuration
@EnableAspectJAutoProxy // 自動生成代理對象
public class PointcutBeanConfig {

    // 注入 myBean1
    @Bean("myBean1")
    public MyBean myBean1() {
        return new MyBean("myBean1");
    }

    //  myBean2
    @Bean("myBean2")
    public MyBean myBean2() {
        return new MyBean("myBean2");
    }

    // 注入切面
    @Bean("pointcutBean")
    public PointcutBean pointcutBean() {
        return new PointcutBean();
    }

    public static void main(String[] args) {
        AnnotationConfigApplicationContext context =
                new AnnotationConfigApplicationContext(PointcutBeanConfig.class);
        MyBean myBean1 = context.getBean("myBean1", MyBean.class);
        myBean1.m();
        MyBean myBean2 = context.getBean("myBean2", MyBean.class);
        myBean2.m();
    }

}

觀察下結果

start execution(void com.crab.spring.aop.demo02.aspectj.MyBean.m())
我是myBean1
我是myBean2

myBean1的方法被攔截了。

上面介紹了Spring中10中切點表達式,下面介紹下切點的組合使用和公共切點的抽取。

切點的組合

切點與切點直接支持邏輯邏輯組合操作: && 、||、 !。使用較小的命名組件構建更復雜的切入點表達式是最佳實踐。

同一個類內切點組合

public class CombiningPointcut {

    /**
     * 匹配 com.crab.spring.aop.demo02包及子包下任何類的public方法
     */
    @Pointcut("execution(public * com.crab.spring.aop.demo02..*.*(..))")
    public void publicMethodPc() {
    }

    /**
     * com.crab.spring.aop.demo02.UserService類的所有方法
     */
    @Pointcut("execution(* com.crab.spring.aop.demo02.UserService.*(..))")
    public void serviceMethodPc(){}


    /**
     * 組合的切點
     */
    @Pointcut("publicMethodPc() && serviceMethodPc()")
    public void combiningPc(){

    }
    /**
     * 組合的切點2
     */
    @Pointcut("publicMethodPc() || !serviceMethodPc()")
    public void combiningPc2(){

    }

}

不同類之間切點組合

切點方法的可見性會影響組合但是不影響切點的匹配。

public class CombiningPointcut2 {

    /**
     * com.crab.spring.aop.demo02.UserService類的所有方法
     */
    @Pointcut("execution(* com.crab.spring.aop.demo02.UserService.*(..))")
    public void serviceMethodPc2(){}


    /**
     * 組合的切點,跨類組合
     */
    @Pointcut("com.crab.spring.aop.demo02.aspectj.reuse.CombiningPointcut.publicMethodPc() && serviceMethodPc2()")
    public void combiningPc(){

    }
    /**
     * 組合的切點,跨類組合,由于serviceMethodPc是private, 此處無法組合
     */
    @Pointcut("com.crab.spring.aop.demo02.aspectj.reuse.CombiningPointcut.serviceMethodPc() && serviceMethodPc2()")
    public void combiningPc2(){

    }
}

切點的公用

在使用企業應用程序時,開發人員通常希望從多個方面引用應用程序的模塊和特定的操作集。建議為此目的定義一個捕獲公共切入點表達式的 CommonPointcuts 方面。直接看案例。

不同層的公共切點

/**
 * 公用的切點
 * @author zfd
 * @version v1.0
 * @date 2022/2/7 8:53
 */
public class CommonPointcuts {

    /**
     * web層的通用切點
     */
    @Pointcut("within(com.xyz.myapp.web..*)")
    public void inWebLayer() {}

    @Pointcut("within(com.xyz.myapp.service..*)")
    public void inServiceLayer() {}

    @Pointcut("within(com.xyz.myapp.dao..*)")
    public void inDataAccessLayer() {}

    @Pointcut("execution(* com.xyz.myapp..service.*.*(..))")
    public void businessService() {}

    @Pointcut("execution(* com.xyz.myapp.dao.*.*(..))")
    public void dataAccessOperation() {}
}

程序中可以直接引用這些公共的切點

/**
 * 使用公共的切點
 * @author zfd
 * @version v1.0
 * @date 2022/2/7 8:56
 */
@Aspect
public class UseCommonPointcuts {

    /**
     * 直接使用公共切點
     */
    @Before("com.crab.spring.aop.demo02.aspectj.reuse.CommonPointcuts.inWebLayer()")
    public void before(JoinPoint joinPoint){
        System.out.println("before:" + joinPoint);
    }
}

總結

本文介紹Spring中10種切點表達式,最常用的是execution,同時介紹切點如何組合使用和如何抽取公共的切點。

 

 

 

 

原文
https://www.cnblogs.com/kongbubihai/p/16017046.html

分享到:
標簽:Spring AOP
用戶無頭像

網友整理

注冊時間:

網站:5 個   小程序:0 個  文章:12 篇

  • 51998

    網站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會員

趕快注冊賬號,推廣您的網站吧!
最新入駐小程序

數獨大挑戰2018-06-03

數獨一種數學游戲,玩家需要根據9

答題星2018-06-03

您可以通過答題星輕松地創建試卷

全階人生考試2018-06-03

各種考試題,題庫,初中,高中,大學四六

運動步數有氧達人2018-06-03

記錄運動步數,積累氧氣值。還可偷

每日養生app2018-06-03

每日養生,天天健康

體育訓練成績評定2018-06-03

通用課目體育訓練成績評定