10

我已经开始使用 Mapstruct 将 JPA 实体映射到 DTO。对于基本实体,这很好用。

我的问题:一些实体有延迟加载的集合,包含额外的细节,我不想总是想要获取和映射。作为解决方案,我添加了一个基本超类,其中包含始终映射的所有字段,以及一个包含集合的子类。它们都代表相同的实体,因此它们使用相同的源类。

当我尝试创建一个包含从同一源映射到两种类型的方法的 Mapper 时,我得到一个模棱两可的映射方法错误,即使方法签名(至少返回类型)不同。我会以错误的方式解决这个问题吗?我不能使用相同的源为 DTO 使用子类吗?

编辑: 如果重要的话,我正在使用 mapstruct-jdk8:1.1.0.Final

编辑 2: 下面的示例只是我头顶上的一个示例。当我实际使用代码时,它起作用了。事实证明,我的问题与示例中未包含的内容有关。当我添加映射票证集合的方法时,似乎发生了错误。这可能意味着问题与继承无关(直接?)。我可能缺少一些配置,但我不确定要寻找什么。

简单的例子:

票务实体

public class Ticket {
  private long id;
  private String title;
  private Set<Comment> comments;

  // Getters and setters
}

机票 DTO

public class TicketDTO {
  private long id;
  private String title;

  // Getters and setters
}

带有评论 DTO 的票证

public class TicketWithCommentsDTO extends TicketDTO {
  private List<CommentDTO> comments;


  // Getters and setters
}

票务映射器界面

@Mapper(uses= { CommentMapper.class })
public interface TicketMapper {
  TicketDTO mapToTicketDTO(Ticket ticket);

  List<TicketDTO> mapToTicketDTOList(Collection<Ticket> tickets); // Adding this method or the last method causes the error

  TicketWithCommentsDTO mapToTicketWithCommentsDTO(Ticket ticket);

  List<TicketWithCommentsDTO> MapToTicketWithCommentDTOList(Collection<Ticket> tickets); 
}

评论映射器界面

@Mapper
public interface CommentMapper {
  CommentDTO toCommentDTO(Comment comment);

  List<CommentDTO> toCommentDTOList(Collection<Comment> comments);
}

抛出的错误:

 Ambiguous mapping methods found for mapping collection element to 
 dto.TicketDTO: dto.TicketDTO mapToTicketDTO(model.Ticket ticket), 
 dto.TicketWithCommentsDTO mapToTicketWithCommentsDTO(model.Ticket ticket).
4

2 回答 2

5

好吧,结果证明这是一个简单的修复,它确实是一个缺少的配置问题。缺少的是@IterableMapping注释。

一旦我将其设置elementTargetType为正确的类型,一切都会按预期工作。

正确的映射器代码

@Mapper(uses = { CommentMapper.class })
public interface TicketMapper {
    TicketDTO mapToTicketDTO(Ticket ticket);

    @IterableMapping(elementTargetType = TicketDTO.class)
    List<TicketDTO> mapToTicketDTOList(Collection<Ticket> tickets);

    TicketWithCommentsDTO mapToTicketWithCommentsDTO(Ticket ticket);

    @IterableMapping(elementTargetType = TicketWithCommentsDTO.class)
    List<TicketWithCommentsDTO> mapToTicketWithCommentDTOList(Collection<Ticket> tickets);
}
于 2017-02-02T13:43:21.203 回答
0

您可以通过添加一些后映射方法来进行子类映射,如下所示:

@Mapper(...)
public interface SuperClassMapper {

SubClassMapper SUBCLASS_MAPPER_INSTANCE = Mappers.getMapper(SubClassMapper.class);

// Mappings entity to dto / dto to entity ...

@AfterMapping
default SuperClassDTO toSubClassDTO(SuperClass entity, @MappingTarget SuperClassDTO dto) 
{
    if (entity instanceof SubClass) {
        return SUBCLASS_MAPPER_INSTANCE.toDto((SubClass) entity);
    }
    return dto;
}

您当然应该已经定义了一个 SubClassMapper。

于 2020-06-09T13:51:47.120 回答