MapStruct:在映射之前过滤列表
MapStruct: filter a list before mapping
我在其他地方看到过这个问题,但不是在相同的上下文中,也没有适合我们用例的答案。
假设我在源对象中有一个列表字段:
List<MySourceElement> mySourceList;
和相应的目标字段:
List<MyTargetElement> myTargetList;
我只是希望能够在执行 mySourceList 的标准映射器以映射到 myTargetList 之前,通过 MySourceElement 的某些属性过滤源 mySourceList 中的元素。
假设 MySourceElement 有一个布尔属性 isValid 并且我们的过滤器谓词很简单 (isValid == true),而 MyTargetElement 没有相应的布尔值。
我尝试了很多方法,包括@DecoratedWith 和 qualifiedBy:
@DecoratedWith 也变得如此 complicated/convoluted 对于这样一个简单的用例,使用 Guice 注入,并且此功能也在 MapStruct 文档中列为 jsr330 的实验性。
qualifiedBy 没有工作(即,我无法让 mapstruct 在实现中应用 qualifiedBy 方法)。
qualifiedBy 方法类似于:
@FilterForValid
public List<MySourceElement> filterForValid(List<MySourceElement> mySourceElement) {
... implementation ...
}
我的映射器声明如下:
@Mapping(source = "mySourceList", target = "myTargetList", qualifiedBy = FilterForValid.class)
Target sourceToTarget(Source source);
我想要一个 qualifiedBy 的实现,例如:
target.withMyTargetList(
mySourceListToMyTargetList(filterUtil.filterForValid(source.getMySourceList)));
代替让 qualifiedBy 工作,我很乐意弄清楚如何为此使用 @BeforeMapping,但我不清楚我将如何从文档中做到这一点,特别是因为对于所有意图和目的源对象应该被认为是不可变的。
任何关于以这种方式结合调用映射器进行集合过滤的最简单、首选方法的指导,我们将不胜感激。
有一个请求的功能 mapstruct/mapstruct#1610 允许使用开箱即用的支持来支持这里的类似功能。话虽如此,解决这个问题的一种方法是使用 @Context
注释。你的映射器看起来像:
@Mapper
public interface MyMapper {
Target map(Source source, @Context Predicate<MySourceElement> predicate);
default List<MySourceElement> mapAndFilter(List<MySourceElement> list, @Context Predicate<MySourceElement> predicate) {
List<MySourceElement> newList = new ArrayList<>();
for(MySourceElement el : list) {
if (predicate.test(el)) {
newList.add(map(el));
}
}
return newList;
}
MySourceElement map(MySourceElement el);
}
我在其他地方看到过这个问题,但不是在相同的上下文中,也没有适合我们用例的答案。
假设我在源对象中有一个列表字段:
List<MySourceElement> mySourceList;
和相应的目标字段:
List<MyTargetElement> myTargetList;
我只是希望能够在执行 mySourceList 的标准映射器以映射到 myTargetList 之前,通过 MySourceElement 的某些属性过滤源 mySourceList 中的元素。
假设 MySourceElement 有一个布尔属性 isValid 并且我们的过滤器谓词很简单 (isValid == true),而 MyTargetElement 没有相应的布尔值。
我尝试了很多方法,包括@DecoratedWith 和 qualifiedBy:
@DecoratedWith 也变得如此 complicated/convoluted 对于这样一个简单的用例,使用 Guice 注入,并且此功能也在 MapStruct 文档中列为 jsr330 的实验性。
qualifiedBy 没有工作(即,我无法让 mapstruct 在实现中应用 qualifiedBy 方法)。
qualifiedBy 方法类似于:
@FilterForValid
public List<MySourceElement> filterForValid(List<MySourceElement> mySourceElement) {
... implementation ...
}
我的映射器声明如下:
@Mapping(source = "mySourceList", target = "myTargetList", qualifiedBy = FilterForValid.class)
Target sourceToTarget(Source source);
我想要一个 qualifiedBy 的实现,例如:
target.withMyTargetList(
mySourceListToMyTargetList(filterUtil.filterForValid(source.getMySourceList)));
代替让 qualifiedBy 工作,我很乐意弄清楚如何为此使用 @BeforeMapping,但我不清楚我将如何从文档中做到这一点,特别是因为对于所有意图和目的源对象应该被认为是不可变的。
任何关于以这种方式结合调用映射器进行集合过滤的最简单、首选方法的指导,我们将不胜感激。
有一个请求的功能 mapstruct/mapstruct#1610 允许使用开箱即用的支持来支持这里的类似功能。话虽如此,解决这个问题的一种方法是使用 @Context
注释。你的映射器看起来像:
@Mapper
public interface MyMapper {
Target map(Source source, @Context Predicate<MySourceElement> predicate);
default List<MySourceElement> mapAndFilter(List<MySourceElement> list, @Context Predicate<MySourceElement> predicate) {
List<MySourceElement> newList = new ArrayList<>();
for(MySourceElement el : list) {
if (predicate.test(el)) {
newList.add(map(el));
}
}
return newList;
}
MySourceElement map(MySourceElement el);
}