前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >一个开胃小菜:Entity To DTO For REST API

一个开胃小菜:Entity To DTO For REST API

作者头像
烟雨平生
发布2023-03-07 16:31:56
5460
发布2023-03-07 16:31:56
举报
文章被收录于专栏:数字化之路数字化之路
原文链接:https://www.baeldung.com/entity-to-and-from-dto-for-a-java-spring-application

作者: baeldung

译者:helloworldtang

1. 概览

在本教程中,我们将处理在Spring应用程序的内部实体和被发送到客户端的外部DTO(数据传输对象)之间的转换。

2. ModelMapper

首先,让我们看看用来执行实体-DTO转换的主要类库——ModelMapper。

我们需要在pom.xml中添加这个依赖:

代码语言:javascript
复制
<dependency>
    <groupId>org.modelmapper</groupId>
    <artifactId>modelmapper</artifactId>
    <version>0.7.4</version>
</dependency>

如果需要检查这个库是否有更新的版本, 请点击这里。

然后,我们将在Spring配置中定义ModelMapper bean:

代码语言:javascript
复制
@Bean
public ModelMapper modelMapper() {
    return new ModelMapper();
}

3. DTO

话分两头,接下来让我们来看看本例使用的DTO——PostDto

代码语言:javascript
复制
public class PostDto {
    private static final SimpleDateFormat dateFormat
      = new SimpleDateFormat("yyyy-MM-dd HH:mm");

    private Long id;

    private String title;

    private String url;

    private String date;

    private UserDto user;

    public Date getSubmissionDateConverted(String timezone) throws ParseException {
        dateFormat.setTimeZone(TimeZone.getTimeZone(timezone));
        return dateFormat.parse(this.date);
    }

    public void setSubmissionDate(Date date, String timezone) {
        dateFormat.setTimeZone(TimeZone.getTimeZone(timezone));
        this.date = dateFormat.format(date);
    }

    // standard getters and setters
}

请注意,上面与日期相关的两个方法,它们是用来处理客户端和服务器之间日期数据转换的:

  • getSubmissionDateConverted()方法将日期字符串转换为服务器所在时区中的日期,以便将其用于持久化Post实体
  • setSubmissionDate()方法是将DTO的日期设置为当前用户所在时区的Post日期

4. 服务层

现在让我们看一下服务层的操作——它显然是与实体(而不是DTO)一起工作:

代码语言:javascript
复制
public List<Post> getPostsList(
  int page, int size, String sortDir, String sort) {

    PageRequest pageReq
     = new PageRequest(page, size, Sort.Direction.fromString(sortDir), sort);

    Page<Post> posts = postRepository
      .findByUser(userService.getCurrentUser(), pageReq);
    return posts.getContent();
}

下面让我们来看看服务层上面的控制器层,这才是转换操作实际触发的地方。

5. 控制器层

现在,让我们来看一个标准的控制器,一个暴露Post资源的REST API。

我们将在这里展示一些简单的CRUD操作:创建、更新、获取一条和全部记录。考虑到操作非常简单,并且我们特别感兴趣的是实体-DTO转换方面

代码语言:javascript
复制
@Controller
class PostRestController {

    @Autowired
    private IPostService postService;

    @Autowired
    private IUserService userService;

    @Autowired
    private ModelMapper modelMapper;

    @RequestMapping(method = RequestMethod.GET)
    @ResponseBody
    public List<PostDto> getPosts(...) {
        //...
        List<Post> posts = postService.getPostsList(page, size, sortDir, sort);
        return posts.stream()
          .map(post -> convertToDto(post))
          .collect(Collectors.toList());
    }

    @RequestMapping(method = RequestMethod.POST)
    @ResponseStatus(HttpStatus.CREATED)
    @ResponseBody
    public PostDto createPost(@RequestBody PostDto postDto) {
        Post post = convertToEntity(postDto);
        Post postCreated = postService.createPost(post));
        return convertToDto(postCreated);
    }

    @RequestMapping(value = "/{id}", method = RequestMethod.GET)
    @ResponseBody
    public PostDto getPost(@PathVariable("id") Long id) {
        return convertToDto(postService.getPostById(id));
    }

    @RequestMapping(value = "/{id}", method = RequestMethod.PUT)
    @ResponseStatus(HttpStatus.OK)
    public void updatePost(@RequestBody PostDto postDto) {
        Post post = convertToEntity(postDto);
        postService.updatePost(post);
    }
}

这是我们从Post实体到PostDto的转换:

代码语言:javascript
复制
private PostDto convertToDto(Post post) {
    PostDto postDto = modelMapper.map(post, PostDto.class);
    postDto.setSubmissionDate(post.getSubmissionDate(), 
        userService.getCurrentUser().getPreference().getTimezone());
    return postDto;
}

这是从DTO到实体的转换:

代码语言:javascript
复制
private Post convertToEntity(PostDto postDto) throws ParseException {
    Post post = modelMapper.map(postDto, Post.class);
    post.setSubmissionDate(postDto.getSubmissionDateConverted(
      userService.getCurrentUser().getPreference().getTimezone()));

    if (postDto.getId() != null) {
        Post oldPost = postService.getPostById(postDto.getId());
        post.setRedditID(oldPost.getRedditID());
        post.setSent(oldPost.isSent());
    }
    return post;
}

因此,正如您所看到的,在modelmapper库的帮助下,转换逻辑是快速且简单的——我们使用了modelMapper的map API,并且在不编写任何转换逻辑的情况下完成了数据转换。

6. 单元测试

最后,让我们做一个非常简单的测试,以确保实体和DTO之间的转换可以很好地工作:

代码语言:javascript
复制
public class PostDtoUnitTest {

    private ModelMapper modelMapper = new ModelMapper();

    @Test
    public void whenConvertPostEntityToPostDto_thenCorrect() {
        Post post = new Post();
        post.setId(Long.valueOf(1));
        post.setTitle(randomAlphabetic(6));
        post.setUrl("www.test.com");

        PostDto postDto = modelMapper.map(post, PostDto.class);
        assertEquals(post.getId(), postDto.getId());
        assertEquals(post.getTitle(), postDto.getTitle());
        assertEquals(post.getUrl(), postDto.getUrl());
    }

    @Test
    public void whenConvertPostDtoToPostEntity_thenCorrect() {
        PostDto postDto = new PostDto();
        postDto.setId(Long.valueOf(1));
        postDto.setTitle(randomAlphabetic(6));
        postDto.setUrl("www.test.com");

        Post post = modelMapper.map(postDto, Post.class);
        assertEquals(postDto.getId(), post.getId());
        assertEquals(postDto.getTitle(), post.getTitle());
        assertEquals(postDto.getUrl(), post.getUrl());
    }
}

7. 总结

本文是关于如何在Spring REST API中使用modelmapper库来简化从实体到DTO以及从DTO到实体的转换,而不是重复造轮子。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2018-09-24,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 的数字化之路 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1. 概览
  • 2. ModelMapper
  • 3. DTO
  • 4. 服务层
  • 5. 控制器层
  • 6. 单元测试
  • 7. 总结
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档