我的问题是:JPA merge 在无状态 Web 应用程序中有作用吗?
SO上关于JPA中的merge操作的讨论很多。还有一个great article on the subject这与 JPA 通过更手动的 Do-It-Yourself 过程合并形成对比(您通过实体管理器找到实体并进行更改)。
我的应用程序有一个丰富的域模型(ala 域驱动设计),它使用 @Version 注释以使用乐观锁定。我们还创建了 DTO,作为我们 RESTful Web 服务的一部分通过网络发送。这个 DTO 层的创建还允许我们向客户端发送它需要的一切,而不是它不需要的。
到目前为止,我理解这是一个相当典型的架构。我的问题是关于需要更新(即 HTTP PUT)现有对象的服务方法。在这种情况下,我们有这两种方法 1) JPA Merge 和 2) DIY。
我不明白的是,JPA 合并如何甚至可以被视为处理更新的选项。这是我的想法,我想知道是否有我不明白的地方:
1) 为了正确地从有线 DTO 创建分离的 JPA 实体,必须正确设置版本号……否则会抛出 OptimisticLockException。但是 JPA 规范说:
An entity may access the state of its version field or property or export a method for use by the application to access the version, but must not modify the version value[30]. Only the persistence provider is permitted to set or update the value of the version attribute in the object.
2) 合并不处理双向关系......反向字段总是以空结束。
3) 如果 DTO 中缺少任何字段或数据(由于部分更新),则 JPA 合并将删除这些关系或清空这些字段。 Hibernate 可以处理部分更新,但不能处理 JPA 合并。 DIY 可以处理部分更新。
4) merge 方法要做的第一件事是查询数据库中的实体 ID,因此与 DIY 相比没有性能优势。
5) 在 DYI 更新中,我们加载实体并根据 DTO 进行更改——就此而言,没有调用 merge 或 persist因为 JPA 上下文开箱即用地实现了工作单元模式。
我有这个直觉吗?
编辑:
6) 关于延迟加载关系的合并行为可以 differ amongst providers .
最佳答案
使用 Merge 确实需要您发送和接收实体的完整表示,或者维护服务器端状态。对于琐碎的 CRUD-y 类型操作,它简单方便。我在无状态 Web 应用程序中大量使用它,在这些应用程序中,让客户端看到整个实体没有任何有意义的安全隐患。
但是,如果您已经将操作减少到只传递直接相关的信息,那么您还需要手动编写相应的服务。
请记住,在进行“DIY”更新时,您仍然需要在 DTO 上传递版本号,并手动将其与数据库中的版本号进行比较。否则,如果您使用更简单的合并方法,您将无法获得跨越“用户思考时间”的乐观锁定。
您不能更改提供者创建的实体的版本,但是当您使用 new 关键字创建了自己的实体类实例时,就可以了,预计在上面设置版本。
它将使持久表示与您提供的内存中表示相匹配,这可以包括使事物为空。请记住,当一个对象被合并时,该对象应该被丢弃并替换为合并返回的对象。您不应该合并一个对象然后继续使用它。它的状态未由规范定义。
正确。
很有可能,只要您的 DIY 解决方案也使用实体 ID 而不是任意查询。 (在查询中使用“查找”方法还有其他好处。)
正确。
关于java - JPA 与 DTO 和乐观锁定合并到 RESTful Web 应用程序中?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14141156/
对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl
我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此
我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r
刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R
我有一个这样的哈希数组:[{:foo=>2,:date=>Sat,01Sep2014},{:foo2=>2,:date=>Sat,02Sep2014},{:foo3=>3,:date=>Sat,01Sep2014},{:foo4=>4,:date=>Sat,03Sep2014},{:foo5=>5,:date=>Sat,02Sep2014}]如果:date相同,我想合并哈希值。我对上面数组的期望是:[{:foo=>2,:foo3=>3,:date=>Sat,01Sep2014},{:foo2=>2,:foo5=>5:date=>Sat,02Sep2014},{:foo4=>4,:dat
我正在学习Rails,并阅读了关于乐观锁的内容。我已将类型为integer的lock_version列添加到我的articles表中。但现在每当我第一次尝试更新记录时,我都会收到StaleObjectError异常。这是我的迁移:classAddLockVersionToArticle当我尝试通过Rails控制台更新文章时:article=Article.first=>#我这样做:article.title="newtitle"article.save我明白了:(0.3ms)begintransaction(0.3ms)UPDATE"articles"SET"title"='dwdwd
我正在尝试使用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
我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我