带你了解如何使用Spring基于ProxyFactoryBean创建AOP代理

目录
  • 1 基础
  • 2 JavaBean属性
  • 3 JDK和CGLIB代理
  • 总结

若使用 Spring IoC 容器(ApplicationContext或BeanFactory)作为你的业务对象(你也应该这么做!),你会想使用 Spring AOP FactoryBean的一种。

工厂 bean 引入了中间层,让它创建不同类型的对象。

在Spring创建 AOP 代理的基本方式是使用 org.springframework.aop.framework.ProxyFactoryBean。这可以完全控制pointcuts、使用的任何通知和他们的顺序。但若不需要这样的控制,也有更简单的选择。

1 基础

ProxyFactoryBean,类似其他 Spring 的FactoryBean实现,引入了中间层。

若你定义了名为 foo 的ProxyFactoryBean,则引用 foo 的对象不会看到ProxyFactoryBean实例本身,而是在ProxyFactoryBean中实现的 getObject()创建的对象。该方法创建了一个包装目标对象的 AOP 代理。

使用ProxyFactoryBean或其他 IoC-aware 类创建 AOP 代理的最大好处之一是advices和pointcuts也可以由 IoC 管理。这是一个强大的功能,使某些方法很难用AOP 实现的开启了新途径。例如,advice本身可能引用应用对象(不仅是目标对象,该对象应在任何 AOP 框架中都可用),从而受益于DI提供的所有可插拔性。

2 JavaBean属性

与大多数FactoryBean实现类似, ProxyFactoryBean类本身就是一个JavaBean。

其属性用于:

  • 指定要代理的目标
  • 指定是否使用 CGLIB

一些关键属性是从 org.springframework.aop.framework.ProxyConfig (Spring所有 Aop 代理工厂的父类) 继承的。这些关键属性包括:

  • proxyTargetClass

如果要代理目标类,而不是目标类的接口,则为 true。如果此属性值设置为true,则创建 CGLIB 代理

  • optimize

控制是否将主动优化应用于通过 CGLIB 创建的代理。除非你完全了解相关的 AOP 代理如何处理优化,否则您不应轻率地使用此设置。仅用于 CGLIB 代理,对 JDK 动态代理无影响。

  • frozen

如果代理配置被冻结,则不再允许更改配置。这既是一种轻微的优化,也是在不希望调用者在创建代理后(通过建议的接口)操纵代理时,这些情况是有用的。默认值false,即允许更改(比如添加额外advice)。

  • exposeProxy

确定当前代理是否应在ThreadLocal暴露,以便目标可以访问该代理。如果目标需要获取代理并将暴露的 Proxy 属性设置为true,则目标可以使用 AopContext.当前普罗西 () 方法。

ProxyFactoryBean其他属性包括:

  • proxyInterfaces

字符串接口名称的数组。若不提供此,则使用目标类的 CGLIB 代理

  • interceptorNames

要应用的Advisor、拦截器或其他建议名称的字符串数组。顺序非常重要,首先先到先得。也就是说,列表中的第一个拦截器是能够拦截调用的第一个拦截器。

这些名称是当前工厂中的bean名称,包括来自祖先工厂的bean名称。你不能在这里使用bean引用, 因为这样做会导致

ProxyFactoryBean忽略了推荐的单例设置。

可以用 * 附加拦截器名称。这样做会导致应用所有advisor beans与名称,开始与*应用前的部分。

  • singleton

工厂是否应该返回单例的对象,无论getObject()调用频率如何,几个FactoryBean实现都提供这样的方法。默认值为true。如果你想使用有状态的advice,使用prototype 类型的advices以及false的singleton值。

3 JDK和CGLIB代理

ProxyFactoryBean如何选择为特定目标对象(将代理)创建基于 JDK 的代理或基于 CGLIB 的代理。

ProxyFactoryBean在创建基于 JDK 或 CGLIB 的代理方面的行为在Spring的 1.2.x 版本和 2.0
版本之间发生了变化。ProxyFactoryBean现在在自动检测接口方面表现出与TransactionProxyFactoryBean类类似的语义。

如果要代理的目标对象类别(以下简称目标类)不实现任何接口,则创建基于 CGLIB 的代理。这是最简单的方案,因为 JDK 代理是基于接口的,没有接口意味着 JDK 代理甚至是不可能的。您可以插入目标豆,并通过设置拦截器命名属性来指定拦截器列表。请注意,即使代理工厂豆的代理目标类属性被设置为虚假,也创建基于 CGLIB 的代理。(这样做是没有意义的,最好从豆的定义中删除,因为它充其量是多余的,而且,在最坏的情况下是令人困惑的。

总结

本篇文章就到这里了,希望能给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • Spring温故而知新系列教程之AOP代理

    AOP的概念 AOP:Aspect-Oriented Programming(面向切面编程),维基百科的解释如下:Aspect是一种新的模块化机制,用来描述分散在对象.类或者函数中的横切关注点,从关注点中分离出横切关注点是面向切面的程序设计的核心概念.分离关注点使解决特定领域问题的代码从业务逻辑中独立出来,业务逻辑的代码中不在含有针对特定领域问题的代码的调用,业务逻辑同特定领域问题的关系通过切面来封装.维护,这样原本分散在整个应用程序中的变动就可以很好地管理起来.从AOP的角度,应用可以分为横切

  • Spring基于ProxyFactoryBean创建AOP代理

    Spring 通知类型 通过前面的学习可以知道,通知(Advice)其实就是对目标切入点进行增强的内容,Spring AOP 为通知(Advice)提供了 org.aopalliance.aop.Advice 接口. Spring 通知按照在目标类方法的连接点位置,可以分为以下五种类型,如表 1 所示. 表 1 Spring 通知的 5 种类型 名称 说明 org.springframework.aop.MethodBeforeAdvice(前置通知) 在方法之前自动执行的通知称为前置通知,可以

  • 详解Java反射实现Aop代理

    利用反射生成JDK的动态代理,也就是AOP中的AOP代理,代替目标对象,从而在代码中织入增强. 定义代理接口 由于JDKf动态代理只能为接口创建动态代理,故先定义接口,假定我们需要对数据的Save方法添加事务处理,我们有一个UserDao接口,里面有一个Save方法,代码如下: public interface UserDao { public void save(); } 定义代理实现 下面具体来实现接口定义的Save方法,我们采用下面的代码来实现. public class UserDaoI

  • Spring-AOP自动创建代理之BeanNameAutoProxyCreator实例

    实例 代码已托管到Github-> https://github.com/yangshangwei/SpringMaster 在 Spring-AOP 静态普通方法名匹配切面 案例中,我们通过配置两个ProxyFactoryBean分别为waiter和seller的Bean创建代理对象, 如下 <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.spring

  • Spring AOP代理详细介绍

    Spring AOP代理详细介绍 前言: 一开始我对spring AOP还是属于一知半解的状态,这几天遇到一个问题,加上又查看了一些Spring相关知识,感觉对这个问题有了更深刻的认识.所以写下来分享一下. 我们知道,Spring支持多种AOP方式,Spring自己的基于代理的AOP和AspectJ的基于编织(weaving)的AOP.如果一个类实现了一个或多个接口,那么Spring就会使用默认的JDK动态代理,如果没有实现任何接口,就会使用cglib来代理.当然我们也可以手动改变这些设置.这也

  • 带你了解如何使用Spring基于ProxyFactoryBean创建AOP代理

    目录 1 基础 2 JavaBean属性 3 JDK和CGLIB代理 总结 若使用 Spring IoC 容器(ApplicationContext或BeanFactory)作为你的业务对象(你也应该这么做!),你会想使用 Spring AOP FactoryBean的一种. 工厂 bean 引入了中间层,让它创建不同类型的对象. 在Spring创建 AOP 代理的基本方式是使用 org.springframework.aop.framework.ProxyFactoryBean.这可以完全控制

  • Spring基于AspectJ的AOP开发案例解析

    目录 AspectJ简介 注解开发 环境准备 不同的通知类型 最通知中通过value属性定义切点 入门案列 @Before前置通知 @AfterReturning后置通知 @Around环绕通知 @AfterThrowing 异常抛出通知 @After 最终通知 通过@Pointcut为切点命名 AspectJ的XML方式的AOP开发 使用AspectJ实现AOP 注解方式 XML方式 AspectJ简介 AspectJ是一个基于Java语言的AOP框架 Spring2.0以后新增了对Aspec

  • 浅析Spring基于注解的AOP

    目录 一.准备工作 二.基于注解的AOP之前置通知 三.基于注解的AOP之切入点表达式的语法和重用以及获取连接点的信息 ①切入点表达式的语法 ②获取连接点的信息 ③重用写入点表达式 一.准备工作 ①创建一个Maven工程 ②添加依赖 在IOC所需依赖基础上再加入下面依赖即可: <!-- spring-aspects会帮我们传递过来aspectjweaver --> <dependency> <groupId>org.springframework</groupId

  • Python基于ThreadingTCPServer创建多线程代理的方法示例

    本文实例讲述了Python基于ThreadingTCPServer创建多线程代理的方法.分享给大家供大家参考,具体如下: #coding=utf8 from BaseHTTPServer import BaseHTTPRequestHandler from SocketServer import ThreadingTCPServer import gzip from StringIO import StringIO import logging logging.basicConfig(level

  • Spring基于XML实现Aop

    目录 项目结构 具体步骤 1.创建maven 项目 导入依赖 创建好项目结构 2.写一个TestDao接口 及实现类 3.编写切面类 测试 总结 项目结构 具体步骤 1.创建maven 项目 导入依赖 创建好项目结构 <dependencies> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <version&g

  • Spring基于advisor配置aop过程解析

    1.目标类 package com.gec.target; public class Hadoop { public void eatting() { System.out.println("大象正在吃东西 1"); try { //耗时5秒 Thread.sleep(5000); } catch (InterruptedException e) { e.printStackTrace(); } } } 2.增强类,此类必须要实现增强方位接口 package com.gec.advic

  • Spring中基于xml的AOP的详细步骤

    1.Aop 全程是Aspect Oriented Programming 即面向切面编程,通过预编译方式和运行期动态代理实现程序功能的同一维护的一种技术.Aop是oop的延续,是软件开发中的 一个热点,也是Spring框架中一个重要的内容.是函数式编程的一个衍生范例,利用Aop可以对业务逻辑各个部分进行分割,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用行,提高了开发效率.简单的说就是把我们程序中的重复代码抽取出来,在需要执行的时候,使用动态代理的技术,在不修改源码的基础上已有的方法进

  • spring基于注解配置实现事务控制操作

    目录 spring注解配置实现事务控制 1.导入相关依赖 2.创建spring配置类 3.创建JdbcConfig数据源配置类 4.创建TransactionConfig事务配置类 5.创建jdbcConfig.properties 6.使用事务注解 Spring注解方式的事务实现机制 1.事务的实现机制 AOP动态代理进行方法拦截 事务管理器进行事务提交或回滚 2.注解方式的事务使用注意事项 正确的设置 @Transactional 的 propagation 属性(熟知事务的传播特性) 正确

  • Spring BPP中如何优雅的创建动态代理Bean详解

    v一.前言 本文章所讲并没有基于Aspectj,而是直接通过Cglib以及ProxyFactoryBean去创建代理Bean.通过下面的例子,可以看出Cglib方式创建的代理Bean和ProxyFactoryBean创建的代理Bean的区别. v二.基本测试代码 测试实体类,在BPP中创建BppTestDepBean类型的代理Bean. @Component public static class BppTestBean { @Autowired private BppTestDepBean d

随机推荐