【SpringBoot苍穹外卖】debugDay04
- 游戏开发
- 2025-09-03 02:54:01

一、getById 与 new
我在修改数据时,产生疑问,注释掉是我一开始写得,new对象是答案提供的,我就好奇两者之间区别。
1. 使用 setmealMapper.getById 获取现有对象 Setmeal setmeal = setmealMapper.getById(setmealDTO.getId()); BeanUtils.copyProperties(setmealDTO, setmeal); // 仅覆盖 setmealDTO 中存在的字段 setmealMapper.update(setmeal); 优点:保留原有数据:从数据库中加载的 Setmeal 对象包含所有字段的当前值,未被 setmealDTO 覆盖的字段会保持不变。
适合部分更新:如果只需要更新部分字段(如名称、价格等),而不想影响其他字段(如创建时间、状态等),这种方式更合适。
数据一致性:确保数据库中的其他字段不会被意外覆盖或丢失。
缺点:额外数据库查询:需要先查询数据库,增加一次 IO 操作,可能影响性能。
代码复杂度:如果字段较多,可能需要手动处理某些字段的更新逻辑。
适用场景:需要保留部分字段(如创建时间、状态等)不被修改。
业务逻辑要求基于现有数据进行更新。
2. 使用 new Setmeal() 创建新对象 Setmeal setmeal = new Setmeal(); BeanUtils.copyProperties(setmealDTO, setmeal); // 完全使用 setmealDTO 的数据 setmealMapper.update(setmeal); 优点:
简单直接:不需要额外的数据库查询,代码更简洁。
性能更高:减少一次数据库查询,适合对性能要求较高的场景。
适合完全覆盖:如果业务逻辑要求完全覆盖数据库中的记录,这种方式更合适。
缺点:丢失原有数据:如果 setmealDTO 中没有包含某些字段(如创建时间、状态等),这些字段会被覆盖为默认值(如 null 或 0),可能导致数据丢失。
不适合部分更新:无法保留未被 setmealDTO 覆盖的字段。
适用场景:需要完全覆盖数据库中的记录。
setmealDTO 包含了所有需要更新的字段。
对性能要求较高,且不需要保留原有数据。
总结与建议
如果需要保留部分字段(如创建时间、状态等),建议使用 setmealMapper.getById,确保数据一致性。
如果完全覆盖数据且 setmealDTO 包含所有字段,建议使用 new Setmeal(),代码更简洁且性能更高。
最佳实践
如果不确定哪种方式更合适,可以采用以下策略:
默认使用 setmealMapper.getById,确保数据安全性和一致性。
在性能敏感的场景下,如果确认 setmealDTO 包含所有字段且不需要保留原有数据,可以使用 new Setmeal()。
if (needPartialUpdate) { Setmeal setmeal = setmealMapper.getById(setmealDTO.getId()); BeanUtils.copyProperties(setmealDTO, setmeal); // 部分更新 setmealMapper.update(setmeal); } else { Setmeal setmeal = new Setmeal(); BeanUtils.copyProperties(setmealDTO, setmeal); // 完全覆盖 setmealMapper.update(setmeal); }根据业务需求灵活选择,既能保证数据安全,又能优化性能。
看来我一开始写得也不错,嘻嘻~。
二、这段代码,为什么copy之后,还要update呢? Setmeal setmeal = setmealMapper.getById(setmealDTO.getId()); // 从数据库获取对象 BeanUtils.copyProperties(setmealDTO, setmeal); // 将 DTO 的数据复制到实体对象 setmealMapper.update(setmeal); // 更新数据库 1. setmealMapper.getById从数据库中加载一个 Setmeal 对象。
这个对象是持久化对象,即它与数据库中的一条记录关联。
2. BeanUtils.copyProperties将 setmealDTO 中的数据复制到 setmeal 对象中。
这一步只是修改了内存中的 setmeal 对象,并没有自动同步到数据库。
3. setmealMapper.update将修改后的 setmeal 对象同步到数据库。
这一步才是真正将内存中的修改持久化到数据库中。
为什么需要 update?
内存与数据库是分离的:
Java 中的对象是内存中的数据结构,修改对象并不会自动同步到数据库。
数据库是一个独立的存储系统,需要通过 SQL 语句(如 UPDATE)来修改数据。
BeanUtils.copyProperties 只是内存操作:
它只是将 setmealDTO 的属性值复制到 setmeal 对象中,修改的是内存中的对象状态。
数据库并不知道这些修改,因此需要显式调用 update 方法将修改同步到数据库。
ORM 框架的工作方式:
如果你使用的是 MyBatis 或 JPA 等 ORM 框架,它们并不会自动跟踪对象的变化。
你需要显式调用 update 方法,告诉框架将对象的状态同步到数据库。
类比理解
可以将这个过程类比为编辑文档:
从文件柜中取出文档(getById):你从数据库加载了一个对象。
在文档上修改内容(copyProperties):你在内存中修改了对象。
将修改后的文档放回文件柜(update):你需要将修改后的对象保存回数据库。
如果你不执行第三步,文件柜中的文档(数据库中的数据)就不会被更新。
如果你不想手动调用 update
如果你希望修改对象后自动同步到数据库,可以考虑以下方式:
使用 JPA(Hibernate):
JPA 提供了自动脏检查机制,修改对象后,在事务提交时会自动同步到数据库。
示例:
Setmeal setmeal = setmealRepository.findById(setmealDTO.getId()).orElseThrow(); BeanUtils.copyProperties(setmealDTO, setmeal); // 不需要显式调用 update,事务提交时会自动同步使用 MyBatis 的动态更新:
可以配置 MyBatis 只更新变化的字段,而不是全部字段。
总结
BeanUtils.copyProperties 只是修改了内存中的对象,数据库并不知道这些变化。
必须显式调用 setmealMapper.update 将修改同步到数据库。
如果你希望自动同步,可以考虑使用 JPA 或其他支持自动脏检查的框架。
三、补充
在写day04时发现少写了个通过套餐id查询菜品的接口,在此记录一下。
【SpringBoot苍穹外卖】debugDay04由讯客互联游戏开发栏目发布,感谢您对讯客互联的认可,以及对我们原创作品以及文章的青睐,非常欢迎各位朋友分享到个人网站或者朋友圈,但转载请说明文章出处“【SpringBoot苍穹外卖】debugDay04”