成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Spring創建AOP代理并非只有@Aspect一種方式

開發 前端
ProxyFactoryBean與其他Spring FactoryBean實現一樣,引入了一個間層。簡單說如果你定義了一個名為foo的ProxyFactoryBean,那么引用foo的對象看不到ProxyFactoryBean實例本身,而是由ProxyFactoryBean中的getObject()方法實現創建的對象。

環境:Spring6.1.2


1. 簡介

在Spring項目中,使用@Aspect注解定義切面(Aspect)并創建AOP(面向切面編程)代理是一種常見的做法,它主要用于實現跨多個類和方法的橫切關注點(Cross-cutting Concerns)的模塊化。下面是對使用@Aspect定義切面以及創建AOP代理的示例:

@Aspect
public class LogAspect {
  // 定義切點
  @Pointcut("execution(* com.pack..*.*(..))")
  public void log() {
  }
  // 前置通知
  @Before("log()")
  public void beforeLog() {
    System.out.println("記錄日志Before...");
  }
  // 后置通知
  @After("log()")
  public void afterLog() {
    System.out.println("記錄日志After");
  }
  // 異常通知
  @AfterThrowing(pointcut = "log()", throwing = "tx")
  public void ex(Throwable tx) {
    System.err.println("發生異常: " + tx.getMessage()) ;
  }
  // 環繞通知
  @Around("log() && args(name)")
  public Object around(ProceedingJoinPoint pjp, String name) throws Throwable {
    System.out.println("log before...") ;
    System.out.println("name = " + name) ;
    Object ret = pjp.proceed() ;
    System.out.println("log after...") ;
    return ret ;
  }
}

以上是一個簡單的異常通知切面定義。在實際工作中絕大多數情況下都是通過上面的方式操作。

但是在某些場景下,你可能需要更細粒度的控制來創建代理對象,比如根據特定條件動態決定是否創建代理、自定義代理的創建過程或調整代理的行為。這時,使用ProxyFactoryBeanProxyFactory可以提供更大的靈活性。ProxyFactoryBean主要用于在Spring容器中配置和創建代理對象,而ProxyFactory則提供了編程式創建代理對象的能力。如果你需要在代碼中動態地創建代理對象,而不是通過Spring容器來管理,那么使用ProxyFactory可能更合適。

接下來將詳細介紹通過ProxyFactoryBean和ProxyFactory創建AOP代理對象。

2. 代理對象創建

2.1 ProxyFactoryBean創建代理

該類提供了對切入點、任何適用的建議及其順序的完全控制。然而,如果您不需要這樣的控制,也可以選擇更簡單的選項。

ProxyFactoryBean與其他Spring FactoryBean實現一樣,引入了一個間層。簡單說如果你定義了一個名為foo的ProxyFactoryBean,那么引用foo的對象看不到ProxyFactoryBean實例本身,而是由ProxyFactoryBean中的getObject()方法實現創建的對象。此方法創建一個AOP代理,用于包裝目標對象。

ProxyFactoryBean很多關鍵的屬性繼承自ProxyConfig(Spring中所有aop代理工廠的超類)。這些關鍵屬性結束如下:

ProxyFactoryBean proxy = new ProxyFactoryBean() ;
// 如果要代理的是目標類,而不是目標類的接口,則為True。如果該屬性值設置為true,則創建CGLIB代理
proxy.setProxyTargetClass(false) ;
// 控制是否對通過CGLIB創建的代理應用積極優化。除非您完全理解相關AOP代理如何處理優化,否則不應該輕松地使用此設置。目前僅用于CGLIB代理。它對JDK動態代理沒有影響。
proxy.setOptimize(false) ;
// 如果代理配置被凍結,則不再允許更改配置。無論是作為輕微的優化,還是當您不希望調用者在創建代理后能夠操作代理(通過建議的接口)時,這都是有用的。此屬性的默認值為false,因此允許更改(例如添加額外的通知)。
proxy.setFrozen(false) ;
// 確定是否應該在ThreadLocal中暴露當前代理,以便目標可以訪問它。如果目標需要獲取代理,而exposeProxy屬性被設置為true,那么可以使用AopContext.currentProxy()方法。
proxy.setExposeProxy(false) ;
// 接口名稱的字符串數組。如果沒有提供,則使用目標類的CGLIB代理
proxy.setProxyInterfaces(new Class<?>[] {}) ;
// 要應用的Advisor、攔截器或其他Advice名稱的字符串數組。點菜很重要,先到先得。也就是說,列表中的第一個攔截器是第一個能夠攔截調用的。
proxy.setInterceptorNames("interceptor01") ;
// 不管getObject()方法被調用的頻率如何,工廠是否應該返回一個對象。有幾個FactoryBean實現提供了這樣的方法。默認值為true
proxy.setSingleton(true) ;

以上是對ProxyFactoryBean創建代理對象時的核心配置說明。

完整使用案例如下:

public interface CommonDAO {}
  
public class PersonService {
  public void save() {
    System.out.println("save method invoke...") ;
  }
}
@Configuration
public class AppConfig {
  @Bean
  public MethodInterceptor logInterceptor() {
    return new MethodInterceptor() {
      @Override
      public Object invoke(MethodInvocation invocation) throws Throwable {
        System.out.println("日志記錄...") ;
        return invocation.proceed() ;
      }
    };
  }
  @Bean
  public ProxyFactoryBean personService() throws Exception {
    ProxyFactoryBean proxy = new ProxyFactoryBean() ;
    proxy.setProxyTargetClass(true) ;
    proxy.setTargetSource(new SingletonTargetSource(new PersonService())) ;
    proxy.setProxyInterfaces(new Class<?>[] {CommonDAO.class}) ;
    proxy.setInterceptorNames("logInterceptor") ;
    return proxy ;
  }
}

2.2 ProxyFactory創建代理

用Spring很容易通過編程創建AOP代理。這讓你可以在不依賴Spring IoC的情況下使用Spring AOP。由目標對象實現的接口會自動被代理。如下示例:

public interface CommonDAO {}
public class PersonService {
  public void save() {
    System.out.println("save method invoke...") ;
  }
}


public static void main(String[] args) {
  ProxyFactory factory = new ProxyFactory(new PersonService()) ;
  factory.setProxyTargetClass(true) ;
  // 設置通知類(內部會自動的包裝為Advisor)
  factory.addAdvice(new MethodInterceptor() {
    @Override
    public Object invoke(MethodInvocation invocation) throws Throwable {
      System.out.println("權限控制...") ;
      return invocation.proceed() ;
    }
  });
  factory.addAdvisor(new PointcutAdvisor() {
    @Override
    public Advice getAdvice() {
      return new MethodInterceptor() {
        @Override
        public Object invoke(MethodInvocation invocation) throws Throwable {
          System.out.println("日志記錄...") ;
          return invocation.proceed() ;
        }
      } ;
    }
    @Override
    public Pointcut getPointcut() {
      return new StaticMethodMatcherPointcut() {
        @Override
        public boolean matches(Method method, Class<?> targetClass) {
          return method.getName().equals("save") ;
        }
      } ;
    }
  }) ;
  PersonService ps = (PersonService) factory.getProxy() ;
  ps.save() ;
}

以上是本篇文章的全部內容,希望對你有幫助。

責任編輯:武曉燕 來源: Spring全家桶實戰案例源碼
相關推薦

2023-09-27 08:01:14

數據推送事件

2025-03-17 08:10:00

aviatorSpringJVM

2013-05-22 15:31:07

AOP的CGlib實現

2025-01-15 12:00:00

Java線程編程

2023-07-27 08:14:29

2022-08-18 09:38:02

Spring跨域

2015-05-06 10:05:22

javajava框架spring aop

2022-06-06 15:44:24

大數據數據分析思維模式

2022-02-14 10:30:37

Java方式框架

2023-08-02 10:48:47

SpringBean反射

2019-08-20 14:13:12

工業物聯網IIOT藍牙

2023-01-26 23:46:15

2009-12-25 16:27:30

MODEM接入方式

2013-12-12 17:58:02

網絡虛擬化疊加SDN

2009-02-26 10:29:00

2021-07-30 16:09:18

加密貨幣比特幣貨幣

2020-12-09 10:15:34

Pythonweb代碼

2023-03-07 15:08:57

2022-07-28 11:29:23

數據安全數據令牌化

2020-12-23 10:10:23

Pythonweb代碼
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 成人在线国产 | 国内精品久久久久久影视8 最新黄色在线观看 | 国产美女一区二区 | 久久黄网 | 日韩精品一区二区三区四区 | 国产精品一码二码三码在线 | 天天操夜夜拍 | 一级毛片免费 | 国产1区2区 | 久热久热| 国产区视频在线观看 | 国产在线视频一区 | 日韩黄色av | 伊人激情综合网 | 国产精品污www一区二区三区 | 91精品国产综合久久久久 | 亚洲天堂一区 | 91精品国产99 | 成人高潮片免费视频欧美 | 亚洲国产精品久久久久婷婷老年 | 欧美一区二区三区 | 日韩在线播放视频 | 综合国产 | 老司机深夜福利网站 | 7799精品视频天天看 | 中文字幕国产精品 | 久久国产区 | 日日操操操 | 羞羞视频网站免费观看 | 久久久久久综合 | 青娱乐国产 | 亚洲精品免费在线 | 亚洲精品不卡 | 伦理片97 | 一区二区三区四区在线免费观看 | 日韩av在线一区二区 | 亚洲精品一区二区在线观看 | 全部免费毛片在线播放网站 | 一区二区三区国产 | 久久免费观看一级毛片 | 欧美福利专区 |