java ModelMapper:匹配多个源属性层次结构

p8h8hvxi  于 2024-01-05  发布在  Java
关注(0)|答案(6)|浏览(194)

我无法解决modelMapper错误。你知道问题出在哪里吗?
注意:鉴于java.sql.Time没有非参数构造函数,我没有找到比编写转换器更好的方法

  1. org.modelmapper.ConfigurationException: ModelMapper configuration errors:
  2. 1) The destination property
  3. biz.models.CarWash.setSecondShift()/java.util.Date.setTime() matches
  4. multiple source property hierarchies:
  5. biz.dto.CarWashDTO.getFirstShift()/java.time.LocalTime.getSecond()
  6. biz.dto.CarWashDTO.getSecondShift()/java.time.LocalTime.getSecond()

字符串
错误是由这段代码造成的

  1. @SpringBootTest
  2. @RunWith(SpringRunner.class)
  3. public class CarWashDTO2CarWash {
  4. @Autowired
  5. protected ModelMapper modelMapper;
  6. @Test
  7. public void testCarWashDTO2CarWash_allFiledShouldBeConverted(){
  8. CarWashDTO dto = CarWashDTO.builder()
  9. .name("SomeName")
  10. .address("SomeAddress")
  11. .boxCount(2)
  12. .firstShift(LocalTime.of(9, 0))
  13. .secondShift(LocalTime.of(20, 0))
  14. .phoneNumber("5700876")
  15. .build();
  16. modelMapper.addConverter((Converter<CarWashDTO, CarWash>) mappingContext -> {
  17. CarWashDTO source = mappingContext.getSource();
  18. CarWash destination = mappingContext.getDestination();
  19. destination.setId(source.getId());
  20. destination.setFirstShift(source.getFirstShift() == null ? null : Time.valueOf(source.getFirstShift()));
  21. destination.setSecondShift(source.getSecondShift() == null ? null : Time.valueOf(source.getSecondShift()));
  22. destination.setEnable(true);
  23. destination.setAddress(source.getAddress());
  24. destination.setBoxCount(source.getBoxCount());
  25. destination.setName(source.getName());
  26. destination.setDateOfCreation(source.getDateOfCreation());
  27. return destination;
  28. });
  29. final CarWash entity = modelMapper.map(dto, CarWash.class);
  30. assertNotNull(entity);
  31. assertEquals(2, entity.getBoxCount().intValue());
  32. assertEquals("SomeAddress", entity.getAddress());
  33. assertEquals("SomeName", entity.getName());
  34. }


}
模型Map器bean由下一个配置构建

  1. @Bean
  2. public ModelMapper modelMapper(){
  3. return new ModelMapper();
  4. }


Dto:

  1. public class CarWashDTO {
  2. private Long id;
  3. private String name;
  4. private String address;
  5. private String phoneNumber;
  6. private Integer boxCount;
  7. private LocalTime firstShift;
  8. private LocalTime secondShift;
  9. private LocalDateTime dateOfCreation;
  10. }


实体(firstShift和secondShift的类型为java.sql.Time):

  1. public class CarWash {
  2. private Long id;
  3. private String name;
  4. private String address;
  5. private String phoneNumber;
  6. private Integer boxCount;
  7. private Time firstShift;
  8. private Time secondShift;
  9. private LocalDateTime dateOfCreation;
  10. private Boolean enable;
  11. private Owner owner;
  12. }

ubof19bj

ubof19bj1#

尝试modelMapper.getConfiguration().setMatchingStrategy(MatchingStrategies.STRICT)

vxqlmq5t

vxqlmq5t2#

这解决了我的问题:modelMapper.getConfiguration().setAmbiguityIgnored(true);

dddzy1tm

dddzy1tm3#

您需要在Bean初始化期间借助PropertyMap自定义ModelMapper配置:http://modelmapper.org/user-manual/property-mapping/

  1. @Bean
  2. public ModelMapper modelMapper(){
  3. ModelMapper mm = new ModelMapper();
  4. PropertyMap<CarWashDTO, CarWash> propertyMap = new PropertyMap<CarWashDTO, CarWash> (){
  5. protected void configure() {
  6. map(source.getId()).setId(null);
  7. }
  8. }
  9. mm.addMappings(propertyMap);
  10. return mm;
  11. }

字符串

jdgnovmf

jdgnovmf4#

我不知道当问题被问到的时候,ModelMapper是怎么样的,但是使用转换器应该很简单。而不是为整个类实现转换器,而是将其实现为实际需要转换的类型。比如:

  1. public static Converter<LocalTime, Time> timeConverter = new AbstractConverter<>() {
  2. @Override
  3. protected Time convert(LocalTime source) {
  4. return null == source ? null : Time.valueOf(source);
  5. }
  6. };

字符串
那么它只是为了:

  1. mm.addConverter(timeConverter);


猜猜你是否知道如何使用Spring或EJB将其添加到配置中。

ef1yzkbh

ef1yzkbh5#

这解决了我的问题:

  1. modelMapper.getConfiguration().setAmbiguityIgnored(true);

字符串

zlwx9yxi

zlwx9yxi6#

如果使用MatchingStrategies.STRICTsetAmbiguityIgnored(true)的建议解决方案不是一个选项(例如,当只有一个ModelMapper示例时,它已经执行了许多不同的Map,因此潜在的副作用可能是一个问题),那么还有另一个解决方案。
而不是使用createTypeMap创建TypeMap(或者typeMap,它的工作原理非常微妙,但基本上是相同的),使用emptyTypeMap。这完全禁用了所有隐式Map的构造,这是在类似命名属性的情况下导致那些二义性问题的原因。在此基础上,然后你可以(也必须)显式地应用所有Map。我猜这可以比作只为这个特定的TypeMap启用STRICT匹配。
所以对于洗车的例子,使用fluent界面,它会是这样的:

  1. modelMapper.emptyTypeMap(CarWashDTO.class, CarWash.class)
  2. .addMapping(CarWashDTO::getFirstShift, CarWash::setFirstShift)
  3. .addMapping(CarWashDTO::getSecondShift, CarWash::setSecondShift)
  4. [...]

字符串
我自己还没有尝试过,但是甚至可以 * 只 * 为冲突的属性显式指定Map,而对其余的属性再次使用隐式Map。这将通过使用implicitMappings()终止链来实现。

相关问题