草庐IT

java - JPA/Hibernate - 防止在 PreRemove 处理程序中删除?

coder 2023-08-28 原文

问题标题基本上说明了一切。在 JPA/Hibernate 中是否有可能优雅地防止从数据库中删除实体?我想要的是将实体标记为“隐藏”而不是实际删除它。

我还希望保留 Cascade 语义,这样如果我尝试删除拥有其他实体集合的实体,则拥有实体及其集合中的每个实体都会被标记为除了实现防止删除并将实体标记为隐藏的 @PreRemove 处理程序外,我无需任何额外工作即可隐藏。

这可能吗,还是我需要找出其他方法?

最佳答案

Is it possible in JPA/Hibernate to gracefully prevent the deletion of an entity from the database?

是的,只要您避免使用 EntityManager.remove(entity) 这是可能的。如果您确实使用 EntityManager.remove(),那么 JPA 提供程序将使用相应的 SQL DELETE 语句标记要删除的对象,这意味着一旦您标记要删除的对象,就不可能有优雅的解决方案。

在 Hibernate 中,您可以使用 @SQLDelete and @Where annotations 实现此目的.但是,这在 JPA 中效果不佳,因为已知 EntityManager.find() 会忽略 @Where 注释中指定的过滤器。

因此,仅 JPA 解决方案将涉及在实体类中添加一个标志,即一列,以区分数据库中逻辑删除的实体与“Activity ”实体。您将需要使用适当的查询(JPQL 和 native )来确保逻辑删除的实体在结果集中不可用。您可以使用 @PreUpdate@PrePersist 注释来 Hook 实体生命周期事件,以确保标志在持续和更新事件上更新。同样,您需要确保不会调用 EntityManager.remove 方法。

我会建议使用 @PreRemove 注释来 Hook 为删除实体而触发的生命周期事件,但是使用实体监听器来防止删除充满了麻烦,原因如下:

  • 如果您需要在逻辑意义上防止 SQL DELETE 发生,您将需要在同一事务中保留对象以重新创建它*。唯一的问题是,在 EntityListener 中引用 EntityManager 并通过推断在监听器中调用 EntityManager.persist 并不是一个好的设计决策。理由很简单 - 您可能最终在 EntityListener 中获得不同的 EntityManager 引用,这只会导致您的应用程序出现模糊和困惑的行为。
  • 如果您需要防止事务本身中的SQL DELETE 发生,那么您必须在您的EntityListener 中抛出一个Exception。这通常最终会回滚事务(特别是如果异常是 RuntimeException 或声明为导致回滚的应用程序异常),并且不会提供任何好处,因为整个事务将被回滚。

如果您可以选择使用 EclipseLink 而不是 Hibernate,那么如果您定义一个适当的 DescriptorCustomizer,那么看起来一个优雅的解决方案是可能的。或者使用 AdditionalCriteria注解。这两个似乎都与 EntityManager.removeEntityManager.find 调用配合得很好。但是,您可能仍需要编写 JPQL 或 native 查询来说明逻辑上删除的实体。


* 这在 JPA Wikibook on the topic of cascading Persist 中有概述。 :

if you remove an object to have it deleted, if you then call persist on the object, it will resurrect the object, and it will become persistent again. This may be desired if it is intentional, but the JPA spec also requires this behavior for cascade persist. So if you remove an object, but forget to remove a reference to it from a cascade persist relationship, the remove will be ignored.

关于java - JPA/Hibernate - 防止在 PreRemove 处理程序中删除?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6645994/

有关java - JPA/Hibernate - 防止在 PreRemove 处理程序中删除?的更多相关文章

  1. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

    我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

  2. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  3. ruby - 在 Ruby 中编写命令行实用程序 - 2

    我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

  4. ruby-on-rails - 如何从 format.xml 中删除 <hash></hash> - 2

    我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为

  5. ruby - 我可以使用 Ruby 从 CSV 中删除列吗? - 2

    查看Ruby的CSV库的文档,我非常确定这是可能且简单的。我只需要使用Ruby删除CSV文件的前三列,但我没有成功运行它。 最佳答案 csv_table=CSV.read(file_path_in,:headers=>true)csv_table.delete("header_name")csv_table.to_csv#=>ThenewCSVinstringformat检查CSV::Table文档:http://ruby-doc.org/stdlib-1.9.2/libdoc/csv/rdoc/CSV/Table.html

  6. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

  7. ruby - 无法运行 Rails 2.x 应用程序 - 2

    我尝试运行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

  8. ruby-on-rails - Rails 应用程序中的 Rails : How are you using application_controller. rb 是新手吗? - 2

    刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr

  9. ruby - 我可以使用 aws-sdk-ruby 在 AWS S3 上使用事务性文件删除/上传吗? - 2

    我发现ActiveRecord::Base.transaction在复杂方法中非常有效。我想知道是否可以在如下事务中从AWSS3上传/删除文件:S3Object.transactiondo#writeintofiles#raiseanexceptionend引发异常后,每个操作都应在S3上回滚。S3Object这可能吗?? 最佳答案 虽然S3API具有批量删除功能,但它不支持事务,因为每个删除操作都可以独立于其他操作成功/失败。该API不提供任何批量上传功能(通过PUT或POST),因此每个上传操作都是通过一个独立的API调用完成的

  10. 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/

随机推荐