如何在 Spring Data 中漂亮地更新 JPA 实体?

新手上路,请多包涵

因此,我查看了有关使用 Spring Data 的 JPA 的各种教程,并且在很多情况下都做了不同的处理,我不太确定正确的方法是什么。

假设有以下实体:

 package stackoverflowTest.dao;

import javax.persistence.*;

@Entity
@Table(name = "customers")
public class Customer {

@Id
@GeneratedValue(strategy = GenerationType.AUTO)
@Column(name = "id")
private long id;

@Column(name = "name")
private String name;

public Customer(String name) {
    this.name = name;
}

public Customer() {
}

public long getId() {
    return id;
}

public String getName() {
    return name;
}

public void setName(String name) {
    this.name = name;
}
}

我们还有一个 DTO,它在服务层中检索,然后交给控制器/客户端。

 package stackoverflowTest.dto;

public class CustomerDto {

private long id;
private String name;

public CustomerDto(long id, String name) {
    this.id = id;
    this.name = name;
}

public long getId() {
    return id;
}

public void setId(long id) {
    this.id = id;
}

public String getName() {
    return name;
}

public void setName(String name) {
    this.name = name;
}
}

所以现在假设客户想要在 webui 中更改他的名字 - 然后会有一些控制器操作,其中将有更新的 DTO 与旧 ID 和新名称。

现在我必须将这个更新的 DTO 保存到数据库中。

不幸的是,目前没有办法更新现有客户(除了删除数据库中的条目并使用新的自动生成的 id 创建一个新的客户)

然而,由于这是不可行的(特别是考虑到这样一个实体可能有数百个关系) - 所以我想到了两个直接的解决方案:

  1. 在 Customer 类中为 id 创建一个设置器 - 从而允许设置 id 然后通过相应的存储库保存 Customer 对象。

或者

  1. 将 id 字段添加到构造函数中,每当您想要更新客户时,您总是使用旧 id 创建一个新对象,但其他字段的新值(在这种情况下只有名称)

所以我的问题是是否有一般规则如何做到这一点?我解释的两种方法的缺点可能是什么?

原文由 Lukas Makor 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 549
2 个回答

比@Tanjim Rahman 回答更好,您可以使用 Spring Data JPA 使用方法 T getOne(ID id)

 Customer customerToUpdate = customerRepository.getOne(id);
customerToUpdate.setName(customerDto.getName);
customerRepository.save(customerToUpdate);

更好,因为 getOne(ID id) 只为您获取一个引用(代理)对象,而不是从数据库中获取它。在此参考上,您可以设置您想要的内容,并且在 save() 上它只会像您期望的那样执行 SQL UPDATE 语句。相比之下,当您调用 find() 就像在@Tanjim Rahmans 中回答 spring 数据时,JPA 将执行 SQL SELECT 以从数据库中物理地获取实体,当您只是更新时,您不需要。

原文由 Robert Niestroj 发布,翻译遵循 CC BY-SA 3.0 许可协议

在 Spring Data 中,如果你有 ID,你只需定义一个更新查询

  @Repository
  public interface CustomerRepository extends JpaRepository<Customer , Long> {

     @Query("update Customer c set c.name = :name WHERE c.id = :customerId")
     void setCustomerName(@Param("customerId") Long id, @Param("name") String name);

  }

一些解决方案声称使用 Spring 数据并改为执行 JPA oldschool(即使以丢失更新的方式)。

原文由 gorefest 发布,翻译遵循 CC BY-SA 4.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题