如何在Spring Data中精美地更新JPA实体?

2022-08-31 16:04:53

因此,我查看了有关JPA与Spring Data的各种教程,并且在很多情况下都以不同的方式完成,我不太确定正确的方法是什么。

假设有一个 follwing 实体:

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中更改其名称 - 然后会有一些控制器操作,其中将使用旧ID和新名称更新DTO。

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

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

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

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

  1. 将id字段添加到构造函数中,每当您要更新客户时,您始终使用旧id创建一个新对象,但其他字段的新值(在本例中仅为名称)

所以我的问题是,是否有一个一般规则如何做到这一点?也许我解释的2种方法的缺点是什么?


答案 1

更好的是,@Tanjim拉赫曼的答案,你可以使用Spring Data JPA使用方法T getOne(ID id)

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

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


答案 2

在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老派(即使以丢失更新的方式)。


推荐