我使用限定符找到了关于Iterable to Non-Iterable mapping的示例:
https://github.com/mapstruct/mapstruct-examples/tree/master/mapstruct-iterable-to-non-iterable
但是如何使这种映射能够映射嵌套属性(使用点注释)?
E.g。将源对象中集合的第一个元素的字段xyz映射到目标对象上的普通字段?
该示例定义限定符
@Qualifier
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface FirstElement {
}
然后定义自定义映射器
public class MapperUtils {
@FirstElement
public <T> T first(List<T> in) {
if (in != null && !in.isEmpty()) {
return in.get(0);
}
else {
return null;
}
}
}
最后,映射定义为
@Mapping(target = "emailaddress", source = "emails", qualifiedBy = FirstElement.class )
但是,如果我想从电子邮件集合的第一个元素中提取特定字段,例如就像我对代码emails.get(0).getEmailAddress
所做的那样?
例如,我希望写一个这样的映射:
@Mapping(target = "emailaddress", source = "emails[0].emailAddress")
答案 0 :(得分:2)
您只需要更改MapperUtils
public class MapperUtils {
@FirstElement
public String firstEmailAddress(List<Person> in) {
if (in != null && !in.isEmpty()) {
return in.get(0).getEmailAddress();
}
else {
return null;
}
}
}
基本上,Annotated方法的参数应该包含您要映射的Iterable
,并且返回类型应该是您要映射到的Non-Iterable
。
如果您不想为映射创建自定义映射,则可以使用expression
属性。
例如:
@Mapping(target = "emailaddress", expression = "emails != null && !emails.isEmpty() ? emails.get(0).getEmailAddress() : null")
但是,如果出错,请小心使用表达式会导致编译时问题。 MapStruct不会检查表达式的有效性,并按原样使用它。
答案 1 :(得分:0)
另一种选择是在mapper中使用以下行 这使用辅助类来帮助进行转换
@Mapping(target = "emailaddress", qualifiedByName={"helperClass", "emailsToAddress"},
source = "emails")
在@Mapper
部分使用的组件中添加helperclass@Mapper(
componentModel="spring",
uses ={
helperClass.class
},
)
帮助程序类看起来像
@Component
@Named("helperClass")
public class helperClass {
@Named("emailsToAddress")
public String emailsToAddress(List<Email> emails) {
if(emails != null || !emails.isEmpty )
return emails.get(0).getAddress();
else
return null;
}