深入理解 toDto 与 toEntity:结合 Eladmin 框架的最佳实践
在现代软件开发中,尤其是后端开发中,数据传输对象(DTO)和实体对象的转换是一个常见且重要的操作。理解和正确实现这种转换不仅能提高代码的可维护性,还能提升应用的性能和安全性。本文将深入探讨 toDto 和 toEntity 方法,并结合 Eladmin 框架,帮助开发者更好地掌握这一关键技术。
什么是 Eladmin?
Eladmin 是一个基于 Spring Boot 的开源管理后台框架,集成了 Spring Security、JWT、Spring Data JPA、MapStruct 等流行技术。它提供了一整套完整的后台管理解决方案,极大地方便了开发者快速构建后台系统。
什么是 DTO 和 实体对象?
数据传输对象(DTO) 是一种设计模式,用于在不同层(如客户端和服务器端)之间传输数据。DTO 通常是一个简单的 POJO(Plain Old Java Object),仅包含数据,不包含业务逻辑。它的主要目的是携带数据并减少网络流量。
实体对象(Entity) 通常是与数据库表直接映射的对象,包含了数据和业务逻辑。在大多数情况下,实体对象用于持久化数据和执行复杂的业务操作。
为什么需要转换?
在一个典型的应用程序中,实体对象和 DTO 之间的转换是不可避免的。主要原因包括:
- 安全性:直接暴露实体对象可能会泄露敏感信息。
- 分离关注点:DTO 关注数据传输,而实体对象关注业务逻辑。
- 简化客户端代码:客户端不需要知道实体对象的内部结构,只需要关心需要的数据。
- 性能优化:DTO 可以减少不必要的数据传输,提升性能。
实现 toDto 和 toEntity 方法
在 Eladmin 中,使用 MapStruct 进行对象转换非常方便。MapStruct 是一个代码生成器,它简化了 Java bean 类型之间的映射。
示例:User 实体和 UserDTO
首先,我们定义一个 User 实体类:
@Entity
@Table(name = "user")
public class User {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;private String username;private String email;private String password;// Getters and setters
}
然后,定义一个 UserDTO 类:
public class UserDTO {private Long id;private String username;private String email;// Getters and setters
}
在 Eladmin 中,通常会有一个 Mapper 接口来定义实体和 DTO 之间的转换:
import org.mapstruct.Mapper;
import org.mapstruct.ReportingPolicy;
import org.mapstruct.factory.Mappers;@Mapper(componentModel = "spring", unmappedTargetPolicy = ReportingPolicy.IGNORE)
public interface UserMapper {UserMapper INSTANCE = Mappers.getMapper(UserMapper.class);UserDTO toDto(User user);User toEntity(UserDTO userDTO);
}
配置和使用 MapStruct 在 Eladmin 中自动生成转换代码
在 Maven 中添加 MapStruct 依赖:
<dependency><groupId>org.mapstruct</groupId><artifactId>mapstruct</artifactId><version>1.4.2.Final</version>
</dependency>
<dependency><groupId>org.mapstruct</groupId><artifactId>mapstruct-processor</artifactId><version>1.4.2.Final</version><scope>provided</scope>
</dependency>
在 application.yml 中配置 MapStruct:
mapstruct:default-component-model: spring
这样,MapStruct 将自动生成实现类,并将其注册为 Spring Bean。你可以直接在服务类中注入使用:
@Service
public class UserService {@Autowiredprivate UserMapper userMapper;public UserDTO getUserDto(Long userId) {User user = userRepository.findById(userId).orElseThrow(() -> new EntityNotFoundException("User not found"));return userMapper.toDto(user);}public User createUser(UserDTO userDTO) {User user = userMapper.toEntity(userDTO);return userRepository.save(user);}
}
结合实际项目:完整的 CRUD 示例
以下是一个完整的 CRUD 示例,展示了如何在 Eladmin 中使用 toDto 和 toEntity 方法进行对象转换。
Controller
@RestController
@RequestMapping("/api/users")
public class UserController {@Autowiredprivate UserService userService;@GetMapping("/{id}")public ResponseEntity<UserDTO> getUser(@PathVariable Long id) {UserDTO userDTO = userService.getUserDto(id);return ResponseEntity.ok(userDTO);}@PostMappingpublic ResponseEntity<UserDTO> createUser(@RequestBody UserDTO userDTO) {User user = userService.createUser(userDTO);UserDTO createdUserDTO = userMapper.toDto(user);return ResponseEntity.status(HttpStatus.CREATED).body(createdUserDTO);}@PutMapping("/{id}")public ResponseEntity<UserDTO> updateUser(@PathVariable Long id, @RequestBody UserDTO userDTO) {User updatedUser = userService.updateUser(id, userDTO);UserDTO updatedUserDTO = userMapper.toDto(updatedUser);return ResponseEntity.ok(updatedUserDTO);}@DeleteMapping("/{id}")public ResponseEntity<Void> deleteUser(@PathVariable Long id) {userService.deleteUser(id);return ResponseEntity.noContent().build();}
}
Service
@Service
public class UserService {@Autowiredprivate UserRepository userRepository;@Autowiredprivate UserMapper userMapper;public UserDTO getUserDto(Long userId) {User user = userRepository.findById(userId).orElseThrow(() -> new EntityNotFoundException("User not found"));return userMapper.toDto(user);}public User createUser(UserDTO userDTO) {User user = userMapper.toEntity(userDTO);return userRepository.save(user);}public User updateUser(Long id, UserDTO userDTO) {User user = userRepository.findById(id).orElseThrow(() -> new EntityNotFoundException("User not found"));user.setUsername(userDTO.getUsername());user.setEmail(userDTO.getEmail());return userRepository.save(user);}public void deleteUser(Long id) {userRepository.deleteById(id);}
}
结论
DTO 和实体对象之间的转换在现代应用开发中至关重要。通过实现 toDto 和 toEntity 方法,或者使用自动化工具如 MapStruct,我们可以简化这一过程,提高代码的可维护性和性能。在 Eladmin 框架中,结合 Spring Boot 和 MapStruct,转换操作变得更加简洁和高效。
希望本文能帮助你更好地理解和掌握 DTO 和实体对象之间的转换技术,并在实际项目中灵活应用。
相关文章:
深入理解 toDto 与 toEntity:结合 Eladmin 框架的最佳实践
在现代软件开发中,尤其是后端开发中,数据传输对象(DTO)和实体对象的转换是一个常见且重要的操作。理解和正确实现这种转换不仅能提高代码的可维护性,还能提升应用的性能和安全性。本文将深入探讨 toDto 和 toEntity 方…...
基于区块链的供应链应用开发
区块链的供应链溯源应用开发 一 、环境准备 (1)更新镜像源 apt update(2)安装(openssl、jdk、git) apt -y install openssl default-jdk git(3)配置JAVA_HOME环境变量 echo “export JAVA_HOME=/usr/lib/jvm/java-11-openjdk-amd64/” >> /etc/profilesource /etc…...
获取GORM执行时的sql字符串
示例: import "log" func GetDetail(tx *gorm.DB,id int)(data any,err error){var query tx.Session(&gorm.Session{DryRun: true})err query.Where("id ?", id).First(&res).Errorif err!nil{zap.L().Error("get detail er…...
Linux系统使用Docker安装RStudio服务并实现任意浏览器远程访问
文章目录 前言1. 安装RStudio Server2. 本地访问3. Linux 安装cpolar4. 配置RStudio server公网访问地址5. 公网远程访问RStudio6. 固定RStudio公网地址 前言 RStudio Server 使你能够在 Linux 服务器上运行你所熟悉和喜爱的 RStudio IDE,并通过 Web 浏览器进行访问…...
【原创】springboot+mysql法律咨询网设计与实现
个人主页:程序猿小小杨 个人简介:从事开发多年,Java、Php、Python、前端开发均有涉猎 博客内容:Java项目实战、项目演示、技术分享 文末有作者名片,希望和大家一起共同进步,你只管努力,剩下的交…...
Vue 应用实例的关键方法与配置案例二
目录 createApp createSSRApp app.mount app.unmount app.component app.directive Vue3.X自定义全局指令 Vue2.X自定义全局指令 app.use app.mixin 非 VIP 用户能够免费下载博文资源 createApp 详见上一章节:Vue 应用实例的关键方法与配置案例一-CSDN博客 createSS…...
Java面试题--JVM大厂篇之破解 JVM 性能瓶颈:实战优化策略大全
目录 引言: 正文: 1. 常见的JVM性能问题 频繁的GC导致应用暂停 内存泄漏导致的内存不足 线程争用导致的CPU利用率过高 类加载问题导致的启动时间过长 2. 优化策略大全 2.1 代码层面的优化 2.1.1 避免不必要的对象创建 2.1.2 优化数据结构的选择 2.1.3 使用并发工具…...
Apache Curator 创建节点时,如果节点存储就会抛出异常吗?
在Apache Curator中,创建节点时,如果该节点已经存在,并且你的代码没有正确处理这种情况,那么会抛出NodeExistsException异常。这是ZooKeeper客户端库(包括Curator)的常见行为,因为它需要确保Zoo…...
【食物链】
题目 代码 #include<bits/stdc.h> using namespace std; const int N 5e410; int n, k; int p[N], d[N]; int find(int x) {if(p[x] ! x){int root find(p[x]);d[x] d[p[x]];p[x] root;}return p[x]; } int main() {ios::sync_with_stdio(0);cin.tie(0);cout.tie(0)…...
【RN】实现markdown文本简单解析
需求 支持文本插入,比如 xxx {product_name} xxx ,如果提供了product_name变量的值为feedback,则可以渲染出 xxx feedback xxx。支持链接解析,比如 [baidu](https://www.baidu.com/),可以直接渲染成超链接的形式。支持…...
webpack plugin
webpack plugin webpack完成的复杂炫酷的功能依赖于插件机制,webpack的插件机制依赖于核心的库, tapable tapable是一个类似于nodejs的eventEmitter的库, 主要是控制钩子函数的发布喝定于,当时,tapable提供您的hook机…...
【busybox记录】【shell指令】date
目录 内容来源: 【GUN】【date】指令介绍 【busybox】【date】指令介绍 【linux】【date】指令介绍 使用示例: 打印前天的日期: 打印三个月零一天后的日期: 打印当年圣诞节的年数: 打印当前的全月名称和月的日期: 要打印一个没有前导零的日期&…...
同态加密和SEAL库的介绍(八)性能
本篇会对比三种加密方案,同时每种方案配置三种参数。即九种情况下的各个操作的性能差异,为大家选择合适的方案和合适的参数提供参考。表格中所有时长的单位均为微妙,即 。 当然数据量比较大,为了方便大家查找,…...
华为OD-D卷数的分解
给定一个正整数n,如果能够分解为m(m > 1)个连续正整数之和,请输出所有分解中,m最小的分解。 如果给定整数无法分解为连续正整数,则输出字符串"N"。 输入描述: 输入数据为一整数,范围为(1, 2^3…...
rk3588 low_delay_net_display注意事项
low_delay_net_display例子默认只支持YUV420和RGB888,如果需要支持YUV422,请添加下面部分: rk3588_nvr/build/app/low_delay_net_display$ git diff v4l2HdmiRX.cpp diff --git a/app/low_delay_net_display/v4l2HdmiRX.cpp b/app/low_delay_net_displa…...
Spring Boot 快速入门样例【后端 3】
Spring Boot 入门:从零到一构建你的第一个应用 Spring Boot 作为一个流行的Java框架,以其“习惯优于配置”的理念极大地简化了Spring应用的开发和部署过程。本文将带你一步步创建一个简单的Spring Boot应用,从环境准备到项目创建,…...
Linux云计算 |【第二阶段】NETWORK-DAY2
主要内容: VLAN技术、TRUNK模式、链路聚合、路由器 一、VLAN技术应用 广播域指接受同样广播消息的节点的集合,如在该集合中的任何一个节点传输一个广播帧,则所有其它能收到这个帧的节点都被认为是该广播帧的一部分; 交换机的所有…...
Java面试题(基础篇)③
目录 一, 与 equals 的区别? 二,接口和抽象类的区别? 三,请说出几个常见的异常? 四,请问你对Java 反射有了解吗? 五,浅拷贝和深拷贝区别? 一,…...
Qt动态调用 - QMetaObject::invokeMethod
QMetaObject::invokeMethod 动态调用是 Qt 的元对象系统的一项强大功能,它允许在运行时通过名称调用槽函数、信号和普通成员函数。 这种能力对于构建灵活和可扩展的应用程序非常有用,比如插件系统或脚本接口。 动态调用方法 Qt 提供了 QMetaObject::i…...
html+css+js网页设计 星享咖啡6个页面(带js) ui还原度90%
htmlcssjs网页设计 星享咖啡6个页面(带js) ui还原度90% 网页作品代码简单,可使用任意HTML编辑软件(如:Dreamweaver、HBuilder、Vscode 、Sublime 、Webstorm、Text 、Notepad 等任意html编辑软件进行运行及修改编辑等…...
G-Helper终极指南:如何用免费开源工具完美控制你的华硕游戏本
G-Helper终极指南:如何用免费开源工具完美控制你的华硕游戏本 【免费下载链接】g-helper Lightweight, open-source control tool for ASUS laptops and ROG Ally. Manage performance modes, fans, GPU, battery, and RGB lighting across Zephyrus, Flow, TUF, St…...
别再做老好人了,优秀PM都有攻击性!
在职场中,“老好人”似乎是一个自带“善意”的标签,但对于项目经理(PM)而言,这三个字往往意味着内耗、妥协与项目失控。很多PM深陷“讨好型人格”的陷阱,怕得罪客户、怕得罪团队、怕得罪领导,凡…...
保姆级教程:用Python搞定数美滑块验证码(含DES加密还原与轨迹模拟)
Python实战:数美滑块验证码全流程破解指南 每次看到那个烦人的滑块验证码,是不是都有种想砸键盘的冲动?特别是当你的爬虫程序在数据采集过程中频繁触发数美验证时,整个项目进度可能都会被拖慢。作为爬虫开发者,我们需要…...
Mochi Diffusion:如何在Mac上实现本地AI图像生成?完整技术指南
Mochi Diffusion:如何在Mac上实现本地AI图像生成?完整技术指南 【免费下载链接】MochiDiffusion Run Stable Diffusion on Mac natively 项目地址: https://gitcode.com/gh_mirrors/mo/MochiDiffusion Mochi Diffusion是一款专为Apple Silicon芯片…...
Pop Shell浮动窗口配置终极指南:如何让特定应用始终保持浮动状态
Pop Shell浮动窗口配置终极指南:如何让特定应用始终保持浮动状态 【免费下载链接】shell Pop!_OS Shell 项目地址: https://gitcode.com/gh_mirrors/sh/shell Pop!_OS Shell(简称Pop Shell)是一款为Linux桌面环境设计的高效窗口管理工…...
窗口像素重构技术:重新定义显示分辨率控制范式
窗口像素重构技术:重新定义显示分辨率控制范式 【免费下载链接】SRWE Simple Runtime Window Editor 项目地址: https://gitcode.com/gh_mirrors/sr/SRWE 问题溯源:窗口分辨率控制的行业痛点解析 在数字内容创作与专业显示领域,窗口分…...
RK3588 Ethernet网络配置与调试实战指南
1. RK3588以太网配置基础入门 第一次拿到RK3588开发板时,很多开发者都会迫不及待地想测试网络功能。作为一款高性能处理器,RK3588通常配备双千兆以太网接口(GMAC0和GMAC1),但在实际使用前需要正确配置才能正常工作。我…...
Diablo Edit2完整指南:掌握暗黑破坏神II角色存档编辑的终极工具
Diablo Edit2完整指南:掌握暗黑破坏神II角色存档编辑的终极工具 【免费下载链接】diablo_edit Diablo II Character editor. 项目地址: https://gitcode.com/gh_mirrors/di/diablo_edit Diablo Edit2是一款功能强大的暗黑破坏神II角色存档编辑器,…...
鸿蒙 ArkUI 技巧实战:把商品分类页的“双栏联动 + 吸顶”做顺手
最近做商城类页面时,我发现一个场景几乎每次都会出现:左边是分类,右边是商品列表。 看起来不复杂,但真写起来很容易翻车——左边点了,右边没滚准;右边手动一滑,左边高亮又跟不上;分组…...
Phi-3-mini-4k-instruct保姆级教学:Ollama Web UI自定义System Prompt与温度调节
Phi-3-mini-4k-instruct保姆级教学:Ollama Web UI自定义System Prompt与温度调节 你是不是已经用Ollama Web UI体验过Phi-3-mini-4k-instruct的文本生成能力了?感觉还不错,但总觉得少了点什么?比如,想让模型扮演一个专…...
