我在其他地方看到过这个问题,但并不完全相同,并且没有适合我们用例的答案。
假设我在源对象中有一个列表字段:
List<MySourceElement> mySourceList;
Run Code Online (Sandbox Code Playgroud)
和相应的目标字段:
List<MyTargetElement> myTargetList;
Run Code Online (Sandbox Code Playgroud)
我只是希望能够在执行 mySourceList 的标准映射器以映射到 myTargetList 之前,通过 MySourceElement 的某些属性来过滤源 mySourceList 中的元素。
假设 MySourceElement 有一个布尔属性 isValid 并且我们的过滤谓词只是 (isValid == true),而 MyTargetElement 没有相应的布尔值。
我尝试了多种方法,包括@DecolatedWith和qualifiedBy:
对于如此简单的用例,@DecolatedWith 与 Guice 注入变得过于复杂/令人费解,并且此功能也在 MapStruct 文档中列为 jsr330 的实验性功能。
qualifiedBy 不起作用(即,我无法让mapstruct 在实现中应用qualifiedBy 方法)。
qualifiedBy 方法类似于:
@FilterForValid
public List<MySourceElement> filterForValid(List<MySourceElement> mySourceElement) {
... implementation ...
}
Run Code Online (Sandbox Code Playgroud)
我的映射器声明如下:
@Mapping(source = "mySourceList", target = "myTargetList", qualifiedBy = FilterForValid.class)
Target sourceToTarget(Source source);
Run Code Online (Sandbox Code Playgroud)
我想要一个qualifiedBy的实现,例如:
target.withMyTargetList(
mySourceListToMyTargetList(filterUtil.filterForValid(source.getMySourceList)));
Run Code Online (Sandbox Code Playgroud)
我很乐意弄清楚如何使用 @BeforeMapping 来实现此目的,而不是让 QualifiedBy 工作,但我不清楚如何从文档中做到这一点,特别是因为对于所有意图和目的,源对象应该被认为是不可变的。
任何有关以这种方式结合调用映射器进行集合过滤的最简单、首选方法的指导将不胜感激。
有一个请求的功能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);
}
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
8112 次 |
| 最近记录: |