如何使用 @patch 使用 jpa 存储库更新单个字段

How to update a single field using jpa repository using @patch

我正在通过 spring 引导开发一个简单的应用程序。我需要限制用户只能更新名称,而不是所有与用户数据相关的文件,但不幸的是,我的代码有一个问题,如果有人以 Json 格式发送数据并更改年龄或任何其他字段都会更新,但正如我所说,我需要用户能够更改唯一名称而不是任何其他字段。我必须提到我正在使用 JPA 存储库和 spring 数据

我的控制器

@RestController
@RequestMapping("/student")
public class StudentController {
    @Autowired
    StudentRepository repository;
    // method i user to only update the name field
    @PatchMapping("/pattt/{id}")

    public ResponseEntity partialUpdateName(
        @RequestBody Student partialUpdate, 
        @PathVariable("id") String id 
    ){
        Student.save(partialUpdate, id);
        return ResponseEntity.ok(repository.save(partialUpdate));
    };
}

JPA 存储库

@Repository
public interface StudentRepository extends JpaRepository<Student, Integer> {}

学生class

@Entity
@Table(name = "student")
public class Student {

    @Id
    @GeneratedValue
    private int id;
    private String name;
    private int age;
    private String emailAddress;

    public Student() {  }
    public Student(int id, String name) {
        this.id = id;
        this.name = name;
    }
    public Student(int id, String name, int age, String emailAddress) {
        this.id = id;
        this.name = name;
        this.age = age;
        this.emailAddress = emailAddress;
    }
    public static void save(Student partialUpdate, String id) {
        partialUpdate.setName(id);
    }
    public void setId(int id) {
        this.id = id;
    }
    public void setName(String name) {
        this.name = name;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public void setEmailAddress(String emailAddress) {
        this.emailAddress = emailAddress;
    }
    public int getId() {
        return id;
    }
    public String getName() {
        return name;
    }
    public int getAge() {
        return age;
    }
    public String getEmailAddress() {
        return emailAddress;
    }
}

未来最好的解决方案是向您的应用程序添加一个 DTO 层,并使用它来映射到您的对象。请参阅下面的示例。

public class StudentDto {
    private String name;

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

然后您可以通过 Mapstruct:

将其映射到您的模型
@Mapper
public abstract class StudentMapper {
    public static final StudentMapper INSTANCE = 
    Mappers.getMapper(StudentMapper.class);
      
    @Mapping
    Student studentDtoToStudent(StudentDto studentDto); 
}

Mapstruct 依赖项:

<dependency>
    <groupId>org.mapstruct</groupId>
    <artifactId>mapstruct-jdk8</artifactId>
    <version>1.3.0.Final</version>
</dependency>
<dependency>
    <groupId>org.mapstruct</groupId>
    <artifactId>mapstruct-processor</artifactId>
    <version>1.3.0.Final</version>
</dependency>

您将能够对外界隐藏您的内部结构。
在你的控制器中:

public ResponseEntity partialUpdateName(
    @RequestBody StudentDto partialUpdate, 
    @PathVariable("id") String id)
{
    Student student = 
    StudentMapper.INSTANCE.studentDtoToStudent(partialUpdate);
}

最后一行会给你一个安全的学生模型,你可以保存它

  1. 快速解决 在您的控制器中:
public ResponseEntity partialUpdateName(
    @RequestBody Student partialUpdate, 
    @PathVariable("id") String id) 
{
    Optional<Student> optionalStudent = repository.findById(id);

    if(optionalStudent.isPresent() && partialUpdate!=null) {
        Student current=optional.get();
        current.setName(partialUpdate.getName());
        return ResponseEntity.ok(repository.save(current));     
     }  
              
     /* return an error */
}