草庐IT

java - 旧的 "@Transactional from within the same class"情况

coder 2023-09-02 原文

原始问题概要: 使用带有 AOP 代理的标准 Spring 事务,不可能从同一类中的非 @Transactional-marked 方法调用 @Transactional-marked 方法并在事务中(特别是由于上述代理)。这在 AspectJ 模式下使用 Spring Transactions 应该是可能的,但它是如何完成的?

编辑: 使用 加载时间 编织的 AspectJ 模式下 Spring 事务的完整纲要:

将以下内容添加到 META-INF/spring/applicationContext.xml :

<tx:annotation-driven mode="aspectj" />

<context:load-time-weaver />

(我假设您已经在应用程序上下文中设置了 AnnotationSessionFactoryBeanHibernateTransactionManager。您可以将 transaction-manager="transactionManager" 作为属性添加到您的 <tx:annotation-driven /> 标记中,但是如果您的事务管理器 bean 的 0x1049 属性的值实际上是 656794“这是多余的,因为“id ”是该属性的默认值。)

添加 transactionManager 。内容如下:
<aspectj>
  <aspects>
    <aspect name="org.springframework.transaction.aspectj.AnnotationTransactionAspect" />
  </aspects>
  <weaver>
    <include within="my.package..*" /><!--Whatever your package space is.-->
  </weaver>
</aspectj>

transactionManagerMETA-INF/aop.xml 添加到您的 aspectjweaver-1.7.0.jar 。我使用 Maven 作为我的构建工具,所以这里是您项目的 spring-aspects-3.1.2.RELEASE.jar 文件的 classpath 声明:
<dependency>
  <groupId>org.aspectj</groupId>
  <artifactId>aspectjweaver</artifactId>
  <version>1.7.0</version>
</dependency>
<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-aspects</artifactId>
  <version>3.1.2.RELEASE</version>
</dependency>
<dependency /> 不需要作为 POM.xml 在您的 spring-instrument-3.1.2.RELEASE.jar 上,但您仍然需要在某处使用它,以便您可以使用 <dependency /> JVM 标志指向它,如下所示:
-javaagent:full\path\of\spring-instrument-3.1.2.RELEASE.jar

我在 Eclipse Juno 中工作,所以为了设置它,我转到了 Window -> Preferences -> Java -> Installed JREs。然后我单击列表框中选中的 JRE,然后单击列表框右侧的“编辑...”按钮。结果弹出窗口中的第三个文本框标记为“默认 VM 参数:”。这是应该输入或复制+粘贴 classpath 标志的地方。

现在是我的实际测试代码类。首先,我的主类 -javaagent :
package my.package;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class TestMain {
  public static void main(String[] args) {
    ApplicationContext applicationContext = new ClassPathXmlApplicationContext("META-INF/spring/applicationContext.xml");
    TestClass testClass = applicationContext.getBean(TestClass.class);
    testClass.nonTransactionalMethod();
  }
}

然后是我的事务类 -javaagent :
package my.package;

import my.package.TestDao;
import my.package.TestObject;
import org.springframework.transaction.annotation.Transactional;

public void TestClass {
  private TestDao testDao;

  public void setTestDao(TestDao testDao) {
    this.testDao = testDao;
  }

  public TestDao getTestDao() {
    return testDao;
  }

  public void nonTransactionalMethod() {
    transactionalMethod();
  }

  @Transactional
  private void transactionalMethod() {
    TestObject testObject = new TestObject();
    testObject.setId(1L);
    testDao.save(testObject);
  }
}

这里的技巧是,如果 TestMain.javaTestClass.java 中的一个字段,则其类将在加载应用程序上下文之前由 TestClass 加载。由于编织是在类的加载时进行的,并且这种编织是由 Spring 通过应用程序上下文完成的,因此它不会被编织,因为在加载应用程序上下文并意识到它之前已经加载了类。
TestMainClassLoader 的进一步细节并不重要。假设它们与 JPA 和 Hibernate 注释连接并使用 Hibernate 进行持久化(因为它们是,并且它们确实如此),并且所有必需的 TestObject 都在应用程序上下文文件中设置。

编辑: 使用 编译时 编织的 AspectJ 模式下 Spring 事务的完整纲要:

将以下内容添加到 TestDao :
<tx:annotation-driven mode="aspectj" />

(我假设您已经在应用程序上下文中设置了 <bean />META-INF/spring/applicationContext.xml。您可以将 AnnotationSessionFactoryBean 作为属性添加到您的 HibernateTransactionManager 标记中,但是如果您的事务管理器 bean 的 0x1049 属性的值实际上是 656794“它是多余的,因为“transaction-manager="transactionManager" ”是该属性的默认值。)

<tx:annotation-driven />id 添加到您的 transactionManager 。我使用 Maven 作为我的构建工具,所以这里是 transactionManager 文件的 spring-aspects-3.1.2.RELEASE.jar 声明:
<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-aspects</artifactId>
  <version>3.1.2.RELEASE</version>
</dependency>
<dependency>
  <groupId>org.aspectj</groupId>
  <artifactId>aspectjrt</artifactId>
  <version>1.7.0</version>
</dependency>

在 Eclipse Juno 中:Help -> Eclipse Marketplace -> 标记为“Find:”的文本框 -> 输入“ajdt” -> 按 [Enter] -> “AspectJ Development Tools (Juno)” -> Install -> Etc。

重新启动 Eclipse(它会让你)后,右键单击您的项目以显示上下文菜单。查看底部附近:配置 -> 转换为 AspectJ 项目。

在您的 aspectjrt-1.7.0.jar 中添加以下 classpath 声明(再次使用 Maven!):
<plugin>
  <groupId>org.codehaus.mojo</groupId>
  <artifactId>aspectj-maven-plugin</artifactId>
  <version>1.4</version>
  <configuration>
    <aspectLibraries>
      <aspectLibrary>
        <groupId>org.springframework</groupId>
        <artifactId>spring-aspects</artifactId>
      </aspectLibrary>
    </aspectLibraries>
  </configuration>
  <executions>
    <execution>
      <goals>
        <goal>compile</goal>
        <goal>test-compile</goal>
      </goals>
    </execution>
  </executions>
</plugin>

替代方法:右键单击您的项目以显示上下文菜单。查看底部附近:AspectJ 工具 -> 配置 AspectJ 构建路径 -> 方面路径选项卡 -> 按“添加外部 JAR...” -> 找到 <dependency /> -> 按“打开” -> 按“确定”。

如果你选择了 Maven 路线,上面的 POM.xml 应该吓坏了。要解决此问题:帮助 -> 安装新软件... -> 按“添加...” -> 在标有“名称:”的文本框中键入您喜欢的任何内容 -> 在标有的文本框中键入或复制+粘贴 <plugin /> “位置:”-> 按“确定”-> 稍等-> 选中“Eclipse AJDT 集成的Maven 集成”旁边的父复选框-> 按“下一步>”-> 安装-> 等。

安装插件并重新启动 Eclipse 后,POM.xml 文件中的错误应该已经消失了。如果没有,请右键单击您的项目以调出上下文菜单:Maven -> 更新项目 -> 按“确定”。

现在是我的实际测试代码类。这次只有一个, full/path/of/spring-aspects-3.1.2.RELEASE.jar :
package my.package;

import my.package.TestDao;
import my.package.TestObject;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.transaction.annotation.Transactional;

public void TestClass {
  private TestDao testDao;

  public void setTestDao(TestDao testDao) {
    this.testDao = testDao;
  }

  public TestDao getTestDao() {
    return testDao;
  }

  public static void main(String[] args) {
    ApplicationContext applicationContext = new ClassPathXmlApplicationContext("META-INF/spring/applicationContext.xml");
    TestClass testClass = applicationContext.getBean(TestClass.class);
    testClass.nonTransactionalMethod();
  }

  public void nonTransactionalMethod() {
    transactionalMethod();
  }

  @Transactional
  private void transactionalMethod() {
    TestObject testObject = new TestObject();
    testObject.setId(1L);
    testDao.save(testObject);
  }
}

这个没有窍门;由于编织发生在编译时,即在类加载和应用程序上下文加载之前,这两件事的顺序不再重要。这意味着所有东西都可以放在同一个类中。在 Eclipse 中,每次点击 Save 时,您的代码都会不断地重新编译(有没有想过它在做什么时说“正在构建工作区:(XX%)”?),所以它已经编织好了,随时可以使用。

就像在加载时间示例中一样:<plugin />http://dist.springsource.org/release/AJDT/configurator/ 的进一步细节并不重要。假设它们与 JPA 和 Hibernate 注释连接并使用 Hibernate 进行持久化(因为它们是,并且它们确实如此),并且所有必需的 POM.xml 都在应用程序上下文文件中设置。

最佳答案

通过阅读您的问题,您并不清楚您的问题在哪里,因此我将简要列出让 AspectJ 拦截您的 @Transactional 所需的条件。方法。

  • <tx:annotation-driven mode="aspectj"/> 在您的 Spring 配置文件中。
  • <context:load-time-weaver/> 以及在您的 Spring 配置文件中。
  • 直接位于类路径中 META-INF 文件夹中的 aop.xml。这个的格式也解释了here .它应该包含处理 @Transactional 的方面定义。注释:<aspect name="org.springframework.transaction.aspectj.AnnotationTransactionAspect"/>
  • 同一个文件中的 weaver 元素也应该有一个 include 子句,告诉它要编织哪些类:<include within="foo.*"/>
  • aspectjrt.jar , aspectjweaver.jar , spring-aspects.jarspring-aop.jar在类路径中
  • 使用标志 -javaagent:/path/to/spring-instrument.jar 启动应用程序(或 spring-agent,在早期版本中被称为)

  • 最后一步可能不是必需的。这是一个非常简单的类,可以使用 InstrumentationLoadTimeWeaver ,但如果不可用,Spring 将尝试使用另一个加载时间编织器。不过,我从未尝试过。

    现在,如果你认为你已经完成了所有步骤但仍然有问题,我可以建议在 weaver 上启用一些选项(在 aop.xml 中定义):
    <weaver options="-XnoInline -Xreweavable -verbose -debug -showWeaveInfo">
    

    这使得编织器输出一堆正在编织的信息。如果您看到正在编织的类,您可以查找您的 TestClass那里。那么你至少有一个起点来继续故障排除。

    关于您的第二次编辑,“这几乎就像编织的速度不够快,无法在类(class)尝试执行之前编织。”,答案是肯定的,这可能会发生。 I experienced a situation like this before .

    我对具体细节有点生疏,但基本上,Spring 将无法编织在创建应用程序上下文之前加载的类。您如何创建应用程序上下文?如果您以编程方式执行此操作,并且该类直接引用了 TestClass ,则可能会出现此问题,因为 TestClass会过早加载。

    不幸的是,我发现调试 AspectJ 简直是 hell 。

    关于java - 旧的 "@Transactional from within the same class"情况,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11615728/

    有关java - 旧的 "@Transactional from within the same class"情况的更多相关文章

    1. ruby-on-rails - rails : "missing partial" when calling 'render' in RSpec test - 2

      我正在尝试测试是否存在表单。我是Rails新手。我的new.html.erb_spec.rb文件的内容是:require'spec_helper'describe"messages/new.html.erb"doit"shouldrendertheform"dorender'/messages/new.html.erb'reponse.shouldhave_form_putting_to(@message)with_submit_buttonendendView本身,new.html.erb,有代码:当我运行rspec时,它失败了:1)messages/new.html.erbshou

    2. ruby-on-rails - 由于 "wkhtmltopdf",PDFKIT 显然无法正常工作 - 2

      我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-

    3. ruby - 检查 "command"的输出应该包含 NilClass 的意外崩溃 - 2

      为了将Cucumber用于命令行脚本,我按照提供的说明安装了arubagem。它在我的Gemfile中,我可以验证是否安装了正确的版本并且我已经包含了require'aruba/cucumber'在'features/env.rb'中为了确保它能正常工作,我写了以下场景:@announceScenario:Testingcucumber/arubaGivenablankslateThentheoutputfrom"ls-la"shouldcontain"drw"假设事情应该失败。它确实失败了,但失败的原因是错误的:@announceScenario:Testingcucumber/ar

    4. ruby - 默认情况下使选项为 false - 2

      这是在Ruby中设置默认值的常用方法:classQuietByDefaultdefinitialize(opts={})@verbose=opts[:verbose]endend这是一个容易落入的陷阱:classVerboseNoMatterWhatdefinitialize(opts={})@verbose=opts[:verbose]||trueendend正确的做法是:classVerboseByDefaultdefinitialize(opts={})@verbose=opts.include?(:verbose)?opts[:verbose]:trueendend编写Verb

    5. ruby - 在没有 sass 引擎的情况下使用 sass 颜色函数 - 2

      我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re

    6. java - 等价于 Java 中的 Ruby Hash - 2

      我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

    7. ruby-on-rails - 迷你测试错误 : "NameError: uninitialized constant" - 2

      我遵循MichaelHartl的“RubyonRails教程:学习Web开发”,并创建了检查用户名和电子邮件长度有效性的测试(名称最多50个字符,电子邮件最多255个字符)。test/helpers/application_helper_test.rb的内容是:require'test_helper'classApplicationHelperTest在运行bundleexecraketest时,所有测试都通过了,但我看到以下消息在最后被标记为错误:ERROR["test_full_title_helper",ApplicationHelperTest,1.820016791]test

    8. ruby-on-rails - 相关表上的范围为 "WHERE ... LIKE" - 2

      我正在尝试从Postgresql表(table1)中获取数据,该表由另一个相关表(property)的字段(table2)过滤。在纯SQL中,我会这样编写查询:SELECT*FROMtable1JOINtable2USING(table2_id)WHEREtable2.propertyLIKE'query%'这工作正常:scope:my_scope,->(query){includes(:table2).where("table2.property":query)}但我真正需要的是使用LIKE运算符进行过滤,而不是严格相等。然而,这是行不通的:scope:my_scope,->(que

    9. 使用 ACL 调用 upload_file 时出现 Ruby S3 "Access Denied"错误 - 2

      我正在尝试编写一个将文件上传到AWS并公开该文件的Ruby脚本。我做了以下事情:s3=Aws::S3::Resource.new(credentials:Aws::Credentials.new(KEY,SECRET),region:'us-west-2')obj=s3.bucket('stg-db').object('key')obj.upload_file(filename)这似乎工作正常,除了该文件不是公开可用的,而且我无法获得它的公共(public)URL。但是当我登录到S3时,我可以正常查看我的文件。为了使其公开可用,我将最后一行更改为obj.upload_file(file

    10. ruby - 安装 Ruby 时遇到问题(无法下载资源 "readline--patch") - 2

      当我尝试安装Ruby时遇到此错误。我试过查看this和this但无济于事➜~brewinstallrubyWarning:YouareusingOSX10.12.Wedonotprovidesupportforthispre-releaseversion.Youmayencounterbuildfailuresorotherbreakages.Pleasecreatepull-requestsinsteadoffilingissues.==>Installingdependenciesforruby:readline,libyaml,makedepend==>Installingrub

    随机推荐