当在Spring Rest Controller中使用PUT请求方法部分更新实体时,我试图区分空值和未提供的值。
考虑以下实体,例如:
@Entity
private class Person {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
/* let's assume the following attributes may be null */
private String firstName;
private String lastName;
/* getters and setters ... */
}
我的个人存储库(Spring Data):
@Repository
public interface PersonRepository extends CrudRepository<Person, Long> {
}
我使用的DTO:
private class PersonDTO {
private String firstName;
private String lastName;
/* getters and setters ... */
}
我的Spring RestController:
@RestController
@RequestMapping("/api/people")
public class PersonController {
@Autowired
private PersonRepository people;
@Transactional
@RequestMapping(path = "/{personId}", method = RequestMethod.PUT)
public ResponseEntity<?> update(
@PathVariable String personId,
@RequestBody PersonDTO dto) {
// get the entity by ID
Person p = people.findOne(personId); // we assume it exists
// update ONLY entity attributes that have been defined
if(/* dto.getFirstName is defined */)
p.setFirstName = dto.getFirstName;
if(/* dto.getLastName is defined */)
p.setLastName = dto.getLastName;
return ResponseEntity.ok(p);
}
}
有丢失财产的请求
{"firstName": "John"}
预期行为:更新firstName=“John”
(保持lastName
不变)。
具有null属性的请求
{"firstName": "John", "lastName": null}
预期行为:更新firstName=“John”
并设置lastName=null
。
我无法区分这两种情况,因为Jackson总是将DTO中的lastName
设置为null
。
注意:我知道REST最佳实践(RFC 6902)建议对部分更新使用补丁而不是PUT,但在我的特定场景中,我需要使用PUT。
还有一个更好的选择,不涉及更改DTO或自定义设置程序。
它包括让Jackson将数据与现有数据对象合并,如下所示:
MyData existingData = ...
ObjectReader readerForUpdating = objectMapper.readerForUpdating(existingData);
MyData mergedData = readerForUpdating.readValue(newData);
newData
中不存在的任何字段都不会覆盖existingData
中的数据,但如果存在字段,则会覆盖该字段,即使该字段包含null
。
演示代码:
ObjectMapper objectMapper = new ObjectMapper();
MyDTO dto = new MyDTO();
dto.setText("text");
dto.setAddress("address");
dto.setCity("city");
String json = "{\"text\": \"patched text\", \"city\": null}";
ObjectReader readerForUpdating = objectMapper.readerForUpdating(dto);
MyDTO merged = readerForUpdating.readValue(json);
结果为{文本:修补文本,地址:地址,城市:空}
在SpringREST控制器中,您需要获取原始JSON数据,而不是让Spring进行反序列化。因此,按如下方式更改endpoint:
@Autowired ObjectMapper objectMapper;
@RequestMapping(path = "/{personId}", method = RequestMethod.PATCH)
public ResponseEntity<?> update(
@PathVariable String personId,
@RequestBody JsonNode jsonNode) {
RequestDto existingData = getExistingDataFromSomewhere();
ObjectReader readerForUpdating = objectMapper.readerForUpdating(existingData);
RequestDTO mergedData = readerForUpdating.readValue(jsonNode);
...
)
按照jackson的作者的建议使用布尔标志。
class PersonDTO {
private String firstName;
private boolean isFirstNameDirty;
public void setFirstName(String firstName){
this.firstName = firstName;
this.isFirstNameDirty = true;
}
public String getFirstName() {
return firstName;
}
public boolean hasFirstName() {
return isFirstNameDirty;
}
}
另一个选项是使用java.util.可选。
import com.fasterxml.jackson.annotation.JsonInclude;
import java.util.Optional;
@JsonInclude(JsonInclude.Include.NON_NULL)
private class PersonDTO {
private Optional<String> firstName;
private Optional<String> lastName;
/* getters and setters ... */
}
如果未设置firstName,则该值为null,@JsonInclude注释将忽略该值。否则,如果在请求对象中隐式设置,firstName将不是null,而是firstName。get()将是。我在浏览解决方案@laffuste时发现它在另一条评论中的链接稍微低了一点(garretwilson最初评论说它不起作用,结果证明它起作用)。
您还可以使用Jackson的ObjectMapper将DTO映射到实体,它将忽略请求对象中未传递的属性:
import com.fasterxml.jackson.databind.ObjectMapper;
class PersonController {
// ...
@Autowired
ObjectMapper objectMapper
@Transactional
@RequestMapping(path = "/{personId}", method = RequestMethod.PUT)
public ResponseEntity<?> update(
@PathVariable String personId,
@RequestBody PersonDTO dto
) {
Person p = people.findOne(personId);
objectMapper.updateValue(p, dto);
personRepository.save(p);
// return ...
}
}
使用可选java.util.验证DTO也有点不同。这里有记录,但我花了一段时间才找到:
// ...
import javax.validation.constraints.NotNull;
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.Pattern;
// ...
private class PersonDTO {
private Optional<@NotNull String> firstName;
private Optional<@NotBlank @Pattern(regexp = "...") String> lastName;
/* getters and setters ... */
}
在这种情况下,firstName可能根本没有被设置,但是如果设置了,如果验证了个人DTO,则可能不会设置为null。
//...
import javax.validation.Valid;
//...
public ResponseEntity<?> update(
@PathVariable String personId,
@RequestBody @Valid PersonDTO dto
) {
// ...
}
还值得一提的是,可选的使用似乎是高度争论的,并且在撰写本文时,Lombok的维护者不支持它(例如,参见这个问题)。这意味着使用龙目岛。数据/龙目岛。类中具有带约束的可选字段的Setter不起作用(它试图创建约束完好无损的setter),因此使用@Setter/@Data会引发异常,因为setter和成员变量都设置了约束。编写没有可选参数的Setter似乎也是更好的形式,例如:
//...
import lombok.Getter;
//...
@Getter
private class PersonDTO {
private Optional<@NotNull String> firstName;
private Optional<@NotBlank @Pattern(regexp = "...") String> lastName;
public void setFirstName(String firstName) {
this.firstName = Optional.ofNullable(firstName);
}
// etc...
}
问题内容: 我试图在Spring Rest Controller中使用PUT请求方法部分更新实体时区分空值和未提供值。 以以下实体为例: 我的人资料库(Spring数据): 我使用的DTO: 我的Spring RestController: 请求缺少属性 预期行为:更新(保留不变)。 具有空属性的请求 预期行为:更新并设置。 我无法区分这两种情况,因为在DTO中,总是由Jackson 设置。 注意
本文向大家介绍区分MySQL中的空值(null)和空字符(''),包括了区分MySQL中的空值(null)和空字符('')的使用技巧和注意事项,需要的朋友参考一下 日常开发中,一般都会涉及到数据库增删改查,那么不可避免会遇到Mysql中的NULL和空字符。 空字符('')和空值(null)表面上看都是空,其实存在一些差异: 定义: 空值(NULL)的长度是NULL,不确定占用了多少存储空间,但是占
如何区分未发送的值和空值?如何检测客户端是否发送了空字段或跳过字段?
我在一个结构中有一个固定大小的缓冲区,我想在它中间复制一些数据。 我现在唯一能看到的就是从开始的部分,加上我想要的,然后在最后加上部分,但是我确信这会导致一两个我想要避免的大副本,我只是需要更新中间的缓冲区。有没有一种简单的方法可以做到这一点而不使用不安全的代码?
问题内容: 让我们考虑以下情况-“文章”文档中有两个字段- content(string)和views(int)。视图字段未建立索引。views字段包含此文章被阅读了多少次的信息。 来自官方文档: 我们还说过文件是不可变的:它们不能更改,只能替换。更新API必须遵守相同的规则。从外部看,似乎我们正在部分更新文档。但是,在内部,更新API仅管理与我们已经描述过的相同的检索- 更改-重新索引过程。 但
我正试图在应用程序处于退出状态时从firebase保存一个通知到redux。 但是我需要在app.js中这样做,当我甚至不在组件上时,我如何分派redux操作呢? 这就是我要做的