优化RESTAPIDTO设计,减少模型冗余技巧
一分耕耘,一分收获!既然打开了这篇文章《优化REST API DTO设计,减少模型冗余方法》,就坚持看下去吧!文中内容包含等等知识点...希望你能在阅读本文后,能真真实实学到知识或者帮你解决心中的疑惑,也欢迎大佬或者新人朋友们多留言评论,多给建议!谢谢!
DTO设计中的冗余挑战
在构建Spring Boot RESTful应用程序时,将HTTP请求体和响应体映射到特定的数据传输对象(DTO)是一种常见的模式。这种做法有助于将业务逻辑与数据传输细节解耦,并提供清晰的数据契约。然而,当请求DTO和响应DTO包含大量相同字段时,往往会引发代码冗余问题。
例如,一个典型的场景是响应DTO需要包含额外的系统元数据,如ID、版本、创建时间、修改时间等,这些数据通常通过继承一个 BaseResponseDTO 来实现。
public abstract class BaseResponseDTO { protected UUID id; protected Integer version; protected Date created; protected Date modified; } public class RequestUserDTO { private String firstName; private String lastName; } public class ResponseUserDTO extends BaseResponseDTO { private String firstName; private String lastName; }
在上述结构中,RequestUserDTO 和 ResponseUserDTO 都包含了 firstName 和 lastName 字段,这导致了显而易见的代码重复。理想情况下,我们希望 ResponseUserDTO 能够同时继承 BaseResponseDTO 和 RequestUserDTO,但这在Java中是不允许的(不支持多重继承)。
传统解决方案及其局限性
为了解决上述冗余问题,开发者可能会尝试以下几种方案:
1. 使用组合模式封装共享字段
一种常见的尝试是引入一个通用的 UserDTO 来封装共享字段,然后让 RequestUserDTO 和 ResponseUserDTO 通过组合的方式引用它:
public abstract class BaseResponseDTO { protected UUID id; protected Integer version; protected Date created; protected Date modified; } public class UserDTO { private String firstName; private String lastName; } public class RequestUserDTO { private UserDTO payload; // 或者直接是 UserDTO user; } public class ResponseUserDTO extends BaseResponseDTO { private UserDTO payload; // 或者直接是 UserDTO user; }
局限性:
- 并未彻底消除代码冗余: 尽管 firstName 和 lastName 被封装到了 UserDTO 中,但 RequestUserDTO 和 ResponseUserDTO 依然需要声明 UserDTO 类型的 payload 字段,这本身也是一种重复。
- 客户端使用不便: 这种设计强制客户端在请求体中将业务数据包装在 payload 字段下(例如 {"payload": {"firstName": "...", "lastName": "..."}}),这增加了API的复杂性和客户端的开发负担。
推荐策略:统一DTO模型
针对上述挑战,当请求和响应的核心业务字段高度一致,且响应的额外字段主要是元数据时,一个更简洁高效的策略是统一请求和响应的DTO模型。
核心思想
该策略的核心在于,创建一个单一的DTO,它既可以用于接收客户端的请求数据,也可以用于向客户端返回响应数据。这个统一的DTO将继承包含通用元数据的基类(如 BaseResponseDTO),从而在响应时提供完整的字段信息。而在请求时,客户端只需发送业务相关的字段,继承自基类的元数据字段即使被发送(通常不会),也会在服务器端被忽略或默认为空,因为它们对请求处理通常不具备业务意义。
实现方式
我们将 RequestUserDTO 和 ResponseUserDTO 合并为一个 UserDTO,并让 UserDTO 直接继承 BaseResponseDTO:
public abstract class BaseResponseDTO { protected UUID id; protected Integer version; protected Date created; protected Date modified; } public class UserDTO extends BaseResponseDTO { private String firstName; private String lastName; }
工作原理:
- 作为请求DTO时: 客户端发送的JSON请求体通常只包含 firstName 和 lastName 等业务字段。BaseResponseDTO 中的字段(如 id, version, created, modified)不会被客户端发送。Spring框架在反序列化时,会根据请求体的内容填充 UserDTO 对象,未提供的字段将保持其默认值(如 null)。服务器端的业务逻辑在处理请求时,只会关注 firstName 和 lastName,而忽略 BaseResponseDTO 中的字段。
- 作为响应DTO时: 服务器端在构建响应时,会创建 UserDTO 实例,并填充 firstName、lastName 以及从 BaseResponseDTO 继承的 id、version、created、modified 等所有相关字段。这些字段随后会被序列化为完整的JSON响应体发送给客户端。
优势分析
采用统一DTO模型的主要优势包括:
- 彻底消除代码冗余: firstName 和 lastName 只定义了一次,极大地减少了重复代码。
- 简化DTO结构: 减少了DTO类的数量,使得项目结构更清晰,更易于理解。
- 提高可维护性: 当业务字段发生变化时,只需修改一个DTO类,降低了维护成本和出错风险。
- API使用友好: 客户端在发送请求时无需额外包装,直接发送业务数据,使API接口更加直观和扁平。
注意事项与适用场景
尽管统一DTO模型具有显著优势,但并非适用于所有情况。在采纳此策略时,需要考虑以下几点:
- 适用场景: 此方法最适用于请求和响应的核心业务字段高度一致,且响应额外字段仅为通用元数据(如ID、时间戳、状态码等)的场景。
- 字段差异性: 如果请求和响应的业务字段存在较大差异(例如,请求只需要部分字段,而响应需要更多或完全不同的字段),则强行统一可能导致DTO臃肿或语义不清。在这种情况下,可以考虑以下替代方案:
- 使用 @JsonView: Spring框架提供的 @JsonView 注解允许在序列化/反序列化时根据不同的视图来包含或排除DTO的字段。这使得同一个DTO可以根据不同的上下文(请求或响应)呈现不同的结构。
- 使用映射工具: 如 MapStruct,可以在不同DTO之间进行字段映射,从而保持请求和响应DTO的独立性,同时避免手动复制字段的繁琐。
- 请求验证: 在请求处理时,如果对传入的字段有严格的验证要求(例如,不允许客户端发送响应特有的元数据字段),则需要在验证逻辑中明确处理。通常,服务器端会忽略请求中与业务逻辑无关的字段,但这需要确保不会引发意外行为或安全漏洞。
- 版本控制: 在API版本迭代中,如果请求和响应的字段演变路径不同,统一DTO可能会增加复杂性。
总结
通过将请求和响应的数据传输对象进行统一,并利用继承机制处理通用元数据,我们能够有效消除REST API设计中的代码冗余,简化DTO结构,并提高系统的可维护性。这种策略在请求和响应的核心业务字段高度重合的场景下尤为有效,为开发者提供了一种简洁而强大的解决方案。然而,在实际应用中,仍需根据具体的业务需求和API复杂性,权衡其适用性,并考虑配合其他技术如 @JsonView 或映射工具来应对更复杂的场景。
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《优化RESTAPIDTO设计,减少模型冗余技巧》文章吧,也可关注golang学习网公众号了解相关技术文章。

- 上一篇
- 智能电视浏览器HTML兼容指南

- 下一篇
- ChatGPT生成数据方法全解析
-
- 文章 · java教程 | 6小时前 |
- Java发送HTML邮件的实用方法
- 367浏览 收藏
-
- 文章 · java教程 | 6小时前 |
- Java处理Protobuf:序列化与反序列化详解
- 214浏览 收藏
-
- 文章 · java教程 | 7小时前 |
- SpringCloudGateway灰度配置全解析
- 362浏览 收藏
-
- 文章 · java教程 | 7小时前 |
- Java连接Cassandra数据库教程
- 385浏览 收藏
-
- 文章 · java教程 | 7小时前 | java 性能监控 网络请求 System.nanoTime() 服务器响应时间
- Java获取服务器响应时间的技巧
- 498浏览 收藏
-
- 文章 · java教程 | 7小时前 |
- Java实现Zookeeper分布式锁详解
- 155浏览 收藏
-
- 文章 · java教程 | 7小时前 |
- JavaUDP通信教程:DatagramSocket使用详解
- 287浏览 收藏
-
- 文章 · java教程 | 7小时前 |
- SpringBoot整合ActiveMQArtemis指南
- 162浏览 收藏
-
- 文章 · java教程 | 7小时前 |
- JavaList增删改查全攻略详解
- 214浏览 收藏
-
- 文章 · java教程 | 8小时前 |
- Java中实例指什么?实例与类关系详解
- 361浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 511次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 498次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- AI歌曲生成器
- AI歌曲生成器,免费在线创作,简单模式快速生成,自定义模式精细控制,多种音乐风格可选,免版税商用,让您轻松创作专属音乐。
- 19次使用
-
- MeloHunt
- MeloHunt是一款强大的免费在线AI音乐生成平台,让您轻松创作原创、高质量的音乐作品。无需专业知识,满足内容创作、影视制作、游戏开发等多种需求。
- 19次使用
-
- 满分语法
- 满分语法是一款免费在线英语语法检查器,助您一键纠正所有英语语法、拼写、标点错误及病句。支持论文、作文、翻译、邮件语法检查与文本润色,并提供详细语法讲解,是英语学习与使用者必备工具。
- 29次使用
-
- 易销AI-专为跨境
- 易销AI是专为跨境电商打造的AI营销神器,提供多语言广告/产品文案高效生成、精准敏感词规避,并配备定制AI角色,助力卖家提升全球市场广告投放效果与回报率。
- 30次使用
-
- WisFile-批量改名
- WisFile是一款免费AI本地工具,专为解决文件命名混乱、归类无序难题。智能识别关键词,AI批量重命名,100%隐私保护,让您的文件井井有条,触手可及。
- 30次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览