我正在使用SpringBoot和Spring REST。 我想了解更新模型属性的HTTP PATCH方法
是否有任何好的教程解释如何使其有效?
答案 0 :(得分:2)
就PATCH
和PUT
而言,POST
方法与@RequestBody
方法没有任何本质区别。挑战在于您传递PATCH请求以及如何在Controller中映射数据。如果使用PATCH
映射到值bean,则必须确定实际设置的内容以及空值的含义。其他选项是限制Map
对一个属性的请求,并在url中指定它或将值映射到HTTP
。
另请参阅Spring MVC PATCH method: partial updates
答案 1 :(得分:2)
我注意到,提供的许多答案都是JSON补丁或不完整的答案。以下是使用功能强大的真实代码的完整说明和示例
首先,PATCH是选择性PUT。您可以使用它来更新对象或对象列表的任意数量的字段。在PUT中,通常会随更新一起发送整个对象。
PATCH / object / 7
{
"objId":7,
"objName": "New name"
}
PUT / object / 7
{
"objId":7,
"objName": "New name",
"objectUpdates": true,
"objectStatus": "ongoing",
"scoring": null,
"objectChildren":[
{
"childId": 1
},
............
}
这使您无需大量端点即可更新记录。例如,在上面,要更新评分,您需要对象/ {id} /评分,然后要更新名称,您需要对象/ {id} /名称。从字面上看,每个项目都有一个端点,或者您需要前端为每个更新发布整个对象。如果您有一个巨大的物体,这可能会花费大量的网络时间或不必要的移动数据。通过该补丁,您可以为1个端点提供移动平台应使用的具有最少对象属性的发送。
这是现实世界中补丁使用的示例:
@ApiOperation(value = "Patch an existing claim with partial update")
@RequestMapping(value = CLAIMS_V1 + "/{claimId}", method = RequestMethod.PATCH)
ResponseEntity<Claim> patchClaim(@PathVariable Long claimId, @RequestBody Map<String, Object> fields) {
// Sanitize and validate the data
if (claimId <= 0 || fields == null || fields.isEmpty() || !fields.get("claimId").equals(claimId)){
return new ResponseEntity<>(HttpStatus.BAD_REQUEST); // 400 Invalid claim object received or invalid id or id does not match object
}
Claim claim = claimService.get(claimId);
// Does the object exist?
if( claim == null){
return new ResponseEntity<>(HttpStatus.NOT_FOUND); // 404 Claim object does not exist
}
// Remove id from request, we don't ever want to change the id.
// This is not necessary, you can just do it to save time on the reflection
// loop used below since we checked the id above
fields.remove("claimId");
fields.forEach((k, v) -> {
// use reflection to get field k on object and set it to value v
// Change Claim.class to whatver your object is: Object.class
Field field = ReflectionUtils.findField(Claim.class, k); // find field in the object class
field.setAccessible(true);
ReflectionUtils.setField(field, claim, v); // set given field for defined object to value V
});
claimService.saveOrUpdate(claim);
return new ResponseEntity<>(claim, HttpStatus.OK);
}
以上内容可能会使某些人感到困惑,因为较新的开发人员通常不会像这样处理反射。基本上,无论您在主体中传递此函数如何,它都会使用给定的ID查找关联的声明,然后仅更新作为键值对传递的字段。
示例正文:
PATCH / claims / 7
{
"claimId":7,
"claimTypeId": 1,
"claimStatus": null
}
以上内容会将ClaimTypeId和claimStatus更新为声明7的给定值,而所有其他值均保持不变。
所以回报会是这样的:
{
"claimId": 7,
"claimSrcAcctId": 12345678,
"claimTypeId": 1,
"claimDescription": "The vehicle is damaged beyond repair",
"claimDateSubmitted": "2019-01-11 17:43:43",
"claimStatus": null,
"claimDateUpdated": "2019-04-09 13:43:07",
"claimAcctAddress": "123 Sesame St, Charlotte, NC 28282",
"claimContactName": "Steve Smith",
"claimContactPhone": "777-555-1111",
"claimContactEmail": "steve.smith@domain.com",
"claimWitness": true,
"claimWitnessFirstName": "Stan",
"claimWitnessLastName": "Smith",
"claimWitnessPhone": "777-777-7777",
"claimDate": "2019-01-11 17:43:43",
"claimDateEnd": "2019-01-11 12:43:43",
"claimInvestigation": null,
"scoring": null
}
如您所见,完整对象将返回,而不更改您想要更改的数据。我知道这里的解释有些重复,我只是想清楚地概述一下。
答案 2 :(得分:-6)
使用 -
创建一个休息模板import org.springframework.http.client.HttpComponentsClientHttpRequestFactory;
RestTemplate rest = new RestTemplate(new HttpComponentsClientHttpRequestFactory());
now make the PATCH call
ResponseEntity<Map<String, Object>> response = rest.exchange(api, HttpMethod.PATCH, request,
responseType);