首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >使用@PATCH方法进行Spring REST部分更新

使用@PATCH方法进行Spring REST部分更新
EN

Stack Overflow用户
提问于 2017-07-20 04:10:16
回答 5查看 34.7K关注 0票数 12

我正在尝试基于以下内容实现Manager实体的部分更新:

实体

代码语言:javascript
运行
复制
public class Manager {
    private int id;
    private String firstname;
    private String lastname;
    private String username;
    private String password;

    // getters and setters omitted
}

控制器中的SaveManager方法

代码语言:javascript
运行
复制
@RequestMapping(value = "/save", method = RequestMethod.PATCH)
public @ResponseBody void saveManager(@RequestBody Manager manager){
    managerService.saveManager(manager);
}

在Dao impl中保存对象管理器。

代码语言:javascript
运行
复制
@Override
public void saveManager(Manager manager) {  
    sessionFactory.getCurrentSession().saveOrUpdate(manager);
}

当我保存对象时,用户名和密码已正确更改,但其他值为空。

所以我需要做的是更新用户名和密码,并保留所有剩余的数据。

EN

回答 5

Stack Overflow用户

回答已采纳

发布于 2017-07-20 04:29:29

您可以编写仅更新特定字段的自定义更新查询:

代码语言:javascript
运行
复制
@Override
public void saveManager(Manager manager) {  
    Query query = sessionFactory.getCurrentSession().createQuery("update Manager set username = :username, password = :password where id = :id");
    query.setParameter("username", manager.getUsername());
    query.setParameter("password", manager.getPassword());
    query.setParameter("id", manager.getId());
    query.executeUpdate();
}
票数 1
EN

Stack Overflow用户

发布于 2017-07-20 04:42:43

如果你真的在使用补丁,那么你应该使用RequestMethod.PATCH,而不是RequestMethod.POST。

补丁程序映射应包含可用于检索要打补丁的Manager对象的id。此外,它应该只包括您想要更改的字段。在您的示例中,您发送的是整个实体,因此您无法辨别实际正在更改的字段( empty是指不使用此字段,还是实际上将其值更改为空)。

也许这样的实现就是您想要的?

代码语言:javascript
运行
复制
@RequestMapping(value = "/manager/{id}", method = RequestMethod.PATCH)
public @ResponseBody void saveManager(@PathVariable Long id, @RequestBody Map<Object, Object> fields) {
    Manager manager = someServiceToLoadManager(id);
    // Map key is field name, v is value
    fields.forEach((k, v) -> {
       // use reflection to get field k on manager and set it to value v
        Field field = ReflectionUtils.findField(Manager.class, k);
        field.setAccessible(true);
        ReflectionUtils.setField(field, manager, v);
    });
    managerService.saveManager(manager);
}
票数 38
EN

Stack Overflow用户

发布于 2017-07-20 13:26:27

有了这个,您就可以修补您的更改

代码语言:javascript
运行
复制
1. Autowire `ObjectMapper` in controller;

2. @PatchMapping("/manager/{id}")
    ResponseEntity<?> saveManager(@RequestBody Map<String, String> manager) {
        Manager toBePatchedManager = objectMapper.convertValue(manager, Manager.class);
        managerService.patch(toBePatchedManager);
    }

3. Create new method `patch` in `ManagerService`

4. Autowire `NullAwareBeanUtilsBean` in `ManagerService`

5. public void patch(Manager toBePatched) {
        Optional<Manager> optionalManager = managerRepository.findOne(toBePatched.getId());
        if (optionalManager.isPresent()) {
            Manager fromDb = optionalManager.get();
            // bean utils will copy non null values from toBePatched to fromDb manager.
            beanUtils.copyProperties(fromDb, toBePatched);
            updateManager(fromDb);
        }
    }

你将不得不扩展BeanUtilsBean来实现非空值行为的复制。

代码语言:javascript
运行
复制
public class NullAwareBeanUtilsBean extends BeanUtilsBean {

    @Override
    public void copyProperty(Object dest, String name, Object value)
            throws IllegalAccessException, InvocationTargetException {
        if (value == null)
            return;
        super.copyProperty(dest, name, value);
    }
}

最后,将NullAwareBeanUtilsBean标记为@Component

或者

NullAwareBeanUtilsBean注册为bean

代码语言:javascript
运行
复制
@Bean
public NullAwareBeanUtilsBean nullAwareBeanUtilsBean() {
    return new NullAwareBeanUtilsBean();
}
票数 19
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/45200142

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档