文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Spring创建AOP代理并非只有@Aspect一种方式

2024-11-30 00:12

关注

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全家桶实战案例源码内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯