正如你所知道的,当你代理一个对象时,比如当你为 Spring/EJB 创建一个具有事务属性的 bean 时,甚至当你使用某些框架创建一个部分模拟时,代理对象不知道这一点,并且内部调用不会被重定向,也不会被拦截...
这就是为什么如果你在 Spring 中做这样的事情:
@Transactionnal
public void doSomething() {
doSomethingInNewTransaction();
doSomethingInNewTransaction();
doSomethingInNewTransaction();
}
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void doSomethingInNewTransaction() {
...
}
当您调用 doSomething 时,您希望在主事务之外还有 3 个新事务,但实际上,由于这个问题,您只能得到一个...
所以我想知道你是如何处理这些问题的......
我实际上处于必须处理复杂事务系统的情况,我认为没有比将服务拆分为许多小服务更好的方法了,这样我就可以确保通过所有代理。 ..
这让我很困扰,因为所有代码都属于同一个功能域,不应该拆分...
我发现这个相关问题的答案很有趣: Spring - @Transactional - What happens in background?
Rob H 说我们可以在服务内部注入(inject) spring 代理,然后调用 proxy.doSomethingInNewTransaction();反而。 这很容易做到,而且很有效,但我真的不喜欢它......
侯云峰说:
So I write my own version of CglibSubclassingInstantiationStrategy and proxy creator so that it will use CGLIB to generate a real subclass which delegates call to its super rather than another instance, which Spring is doing now. So I can freely annotate on any methods(as long as it is not private), and from wherever I call these methods, they will be taken care of. Well, I still have price to pay: 1. I must list all annotations that I want to enable the new CGLIB sub class creation. 2. I can not annotate on a final method since I am now generating subclass, so a final method can not be intercepted.
他所说的“现在哪个 Spring 在做什么”是什么意思?这是否意味着内部事务调用现在被拦截了?
你觉得哪个更好?
当您需要一些事务粒度时,您会拆分类吗? 还是您使用上述的一些解决方法? (请分享)
最佳答案
我将讨论 Spring 和 @Transactional,但该建议也适用于许多其他框架。
这是基于代理方面的固有问题。此处的 spring 文档中对此进行了讨论:
有许多可能的解决方案。
重构您的类以避免绕过代理的自调用调用。
Spring 文档将其描述为 “最佳方法(此处使用的术语最好)”。
这种方法的优点是简单,并且与任何框架都没有联系。但是,它可能不适合事务繁重的代码库,因为您最终会得到许多微不足道的小类。
在类内部获取对代理的引用。
这可以通过注入(inject)代理或使用硬编码的“AopContext.currentProxy()”调用来完成(参见上面的 Spring 文档)。
此方法允许您避免拆分类,但在许多方面否定了使用事务注释的优势。我个人的看法是,这是一种有点丑陋的东西,但丑陋是自包含的,如果使用大量交易,这可能是务实的做法。
切换到使用 AspectJ
由于 AspectJ 不使用代理,因此自调用不是问题
虽然这是一个非常干净的方法 - 它是以引入另一个框架为代价的。我参与了一个大型项目,正是出于这个原因引入了 AspectJ。
根本不要使用@Transactional
重构您的代码以使用手动事务划分 - 可能使用装饰器模式。
一个选项 - 但需要适度重构,引入额外的框架联系并增加复杂性 - 所以可能不是首选选项
我的建议
通常拆分代码是最好的答案,对于分离关注点也是一件好事。但是,如果我有一个严重依赖嵌套事务的框架/应用程序,我会考虑使用 AspectJ 来允许自调用。
关于java - 如何处理 Spring/EJB/Mockito... 代理上的内部调用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8039041/
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
我正在使用Ruby2.1.1和Rails4.1.0.rc1。当执行railsc时,它被锁定了。使用Ctrl-C停止,我得到以下错误日志:~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.2/lib/spring/client/run.rb:47:in`gets':Interruptfrom~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.2/lib/spring/client/run.rb:47:in`verify_server_version'from~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.
我在理解Enumerator.new方法的工作原理时遇到了一些困难。假设文档中的示例:fib=Enumerator.newdo|y|a=b=1loopdoy[1,1,2,3,5,8,13,21,34,55]循环中断条件在哪里,它如何知道循环应该迭代多少次(因为它没有任何明确的中断条件并且看起来像无限循环)? 最佳答案 Enumerator使用Fibers在内部。您的示例等效于:require'fiber'fiber=Fiber.newdoa=b=1loopdoFiber.yieldaa,b=b,a+bendend10.times.m
我想设置一个默认日期,例如实际日期,我该如何设置?还有如何在组合框中设置默认值顺便问一下,date_field_tag和date_field之间有什么区别? 最佳答案 试试这个:将默认日期作为第二个参数传递。youcorrectlysetthedefaultvalueofcomboboxasshowninyourquestion. 关于ruby-on-rails-date_field_tag,如何设置默认日期?[rails上的ruby],我们在StackOverflow上找到一个类似的问
我将我的Rails应用程序部署到OpenShift,它运行良好,但我无法在生产服务器上运行“Rails控制台”。它给了我这个错误。我该如何解决这个问题?我尝试更新rubygems,但它也给出了权限被拒绝的错误,我也无法做到。railsc错误:Warning:You'reusingRubygems1.8.24withSpring.UpgradetoatleastRubygems2.1.0andrun`gempristine--all`forbetterstartupperformance./opt/rh/ruby193/root/usr/share/rubygems/rubygems
我正在尝试从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
我正在尝试编写一个将文件上传到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
如何在ruby中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL
我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www
我需要一些关于TDD概念的帮助。假设我有以下代码defexecute(command)casecommandwhen"c"create_new_characterwhen"i"display_inventoryendenddefcreate_new_character#dostufftocreatenewcharacterenddefdisplay_inventory#dostufftodisplayinventoryend现在我不确定要为什么编写单元测试。如果我为execute方法编写单元测试,那不是几乎涵盖了我对create_new_character和display_invent