gpt4 book ai didi

java - 我可以在同一应用程序上下文中同时使用动态代理和 CGLib 代理吗?

转载 作者:行者123 更新时间:2023-11-30 11:22:41 26 4
gpt4 key购买 nike

我正在为生产 spring batch 项目构建端到端测试框架。我们想使用内存中的嵌入式数据库(在我们的例子中是 hsqldb)进行这些测试,因为它运行得更快并且没有环境限制。由于这些数据库引擎之间的 SQL 语法略有不同,我们想使用 spring AOP 来拦截进入数据库的 sql,并取出/替换不支持的 sql 命令。 (我们已经研究了完全参数化 sql,但我们将不兼容的命令减少到只有一些小东西,例如“(nolock)”提示。)

我为 JDBCTemplate 设置了一个 AOP 方面,它遵循一个非常标准的 aop 模式来记录 sql:

@Pointcut("execution(* org.springframework.jdbc.core.JdbcOperations.*(String, ..))")
public void modifyJdbcOperations() {
}

它与这个 spring 配置设置配合得很好:

<aop:aspectj-autoproxy proxy-target-class="true" >
<aop:include name = "SQLModifierAspect"/>
</aop:aspectj-autoproxy>

<aop:config proxy-target-class="true">
</aop:config>

<bean id="SQLModifierAspect" class="e2e.framework.SQLModifierAspect" />

然后我尝试为构造函数注入(inject)的我们的一个 DAO 对象上的方法设置一个类似的切入点:

@Pointcut("execution(* com.*.dao.DateDao.buildSelectByDateCarryForward())")
public void modifyDateDao() {
}

但是当我使用这个切入点运行 E2E 框架时,我得到:

Caused by: java.lang.IllegalArgumentException: Superclass has no null constructors but no arguments were given
at net.sf.cglib.proxy.Enhancer.emitConstructors(Enhancer.java:721) ~[cglib-2.2.jar:na]

我看到这是因为我已经配置为使用 CGLIB 而不是基于 Spring 动态代理的 AOP。我在这里阅读了差异:http://docs.spring.io/spring/docs/3.0.0.M3/reference/html/ch08s06.html

我无法现实地重构我们的整个生产项目以使用属性 DI 而不是构造函数 DI。如果我切换回 Spring 动态代理,那么 JdbcTemplate AOP 将不起作用:

org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'jdbcSupportBI' defined in class path resource [e2e-overrides.xml]: Initialization of bean failed; nested exception is org.springframework.beans.ConversionNotSupportedException: Failed to convert property value of type '$Proxy51 implementing org.springframework.jdbc.core.JdbcOperations,org.springframework.beans.factory.InitializingBean,org.springframework.aop.SpringProxy,org.springframework.aop.framework.Advised' to required type 'org.springframework.jdbc.core.JdbcTemplate' for property 'jdbcTemplate';

所以我的问题是:我能以某种方式配置 aspectj 以对某些类使用 Spring 动态代理,而对其他类使用 CGLIB 吗?是否有另一种方法可以解决这个问题,只需要最少的重构?

更新:我当前的解决方法是根据需要将 SqlPreparer 类注入(inject)到 Daos 中。我可以让它有一个默认的构造函数。它添加了一个额外的 util 类,该类在生产代码中不执行任何操作,但它不像重构那样具有侵入性:

/**
* Abstracted out of the GenericDao for AOP in the E2E tests
*/
public class SqlPreparer implements ISqlPreparer {

@Override
public String prepareSql(String sql) {
return sql;
}
}

最佳答案

您可以简单地将上下文拆分为多个上下文。一个用于您的 JdbcTemplate,具有通过 CGLIB 代理的方面,一个用于您的 DAO,具有通过 JDK 代理代理的方面。

错误本身是因为 Spring 似乎使用默认的 CGLIB Enhancer 代理行为,使用类的无参数构造函数。由于您的类(class)没有类(class),因此它失败了。您可以重构您的 DataDao 类并将构造函数注入(inject)移动到字段或 setter 注入(inject),而不是使用无参数构造函数。

关于java - 我可以在同一应用程序上下文中同时使用动态代理和 CGLib 代理吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21585052/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com