我将业务对象映射到实体,并且存在实体结构与业务对象不同的情况。
我有userCategories
存储在业务对象中 RecipeBo
作为字符串,因为 BO 不必了解实体的内部结构。这些字符串需要映射到 Recipe
的关系和 RecipeUserCategoryRel
,除此之外,还有另一个字段,userId
的 RecipeBo
需要映射到这个 RecipeUserCategoryRel
也是。
我的方法(有效)是创建一个包装器并手动创建关系,但这看起来像是在修补:
public class BoMapper
{
private final static ModelMapper modelMapper = new ModelMapper();
static
{
modelMapper.addMappings(new IngredientMap());
}
public static void map(Object from, Object to)
{
modelMapper.map(from, to);
if (from instanceof RecipeBo && to instanceof Recipe)
{
RecipeBo recipeBo = (RecipeBo)from;
List<String> userCategories = recipeBo.getUserCategories();
List<RecipeUserCategoryRel> recipeUserCategoryRels = new ArrayList<>();
for (String userCategory : userCategories)
{
recipeUserCategoryRels.add(new RecipeUserCategoryRel(userCategory, recipeBo.getUserId()));
}
Recipe recipe = (Recipe)to;
recipe.setRecipeUserCategoryRels(recipeUserCategoryRels);
}
}
}
对于我在 BoMapper 中所做的事情,是否有更好的方法,例如使用转换器之类的?困难在于映射列表的每个元素并添加 userId
领域也是如此。
最佳答案
问题
这是一个复杂的情况,因为您是从其他层次结构而不是直接从列表中获取 userId。 ModelMapper 会将 List 映射到 List,但如果您不将 ModelMapper 配置为 LOOSE,它将无法工作。
modelMapper.getConfiguration()
.setMatchingStrategy(MatchingStrategies.LOOSE);
无论如何,如果您以这种方式(松散模式)配置 ModelMapper,它将映射列表并放入您的类 RecipeUserCategoryRel 的字符串属性(在这种情况下,例如 userCategory,如果它是一个字符串,并且考虑到 userId 不是一个字符串) 其他(我不太确定)我认为它会是空的。
解决方案
好吧,我认为您的问题的解决方案是创建一个 Converter 并将其添加到您的 ModelMapper 实例:
- RecipeBO(源)-> Recipe(目标)
代码如下:
ModelMapper mapper = new ModelMapper();
Converter<RecipeBO, Recipe> converter = new Converter<RecipeBO,
Recipe>() {
@Override
public Recipe convert(MappingContext<RecipeBO, Recipe> context) {
RecipeBO source = context.getSource();
Recipe destination = new Recipe();
List<String> userCategoryValues = source.getUserCategories();
List<RecipeUserCategoryRel> userCategoryToMap = new ArrayList<RecipeUserCategoryRel>();
for(final String userCategory : userCategoryValues){
userCategoryToMap.add(new RecipeUserCategoryRel(userCategory,source.getUserId()));
}
destination.setRecipeUserCategoryRels(userCategoryToMap);
//... Map other properties if you need
return destination;
}
};
//Option 1
mapper.createTypeMap(RecipeBO.class, Recipe.class).setConverter(converter);
//If you add as a converter directly also works (I don't know which one is better,
//choose above option (createTypeMap + setConverter) or the next (addConverter)
//Option 2 -> mapper.addConverter(converter);
我已经测试过并且有效!!
如果我有下一个食谱:
RecipeBO recipe = new RecipeBO();
recipe.setUserId("1");
String values[] = new String[] { "abc", "klm", "xyz", "pqr" };
List<String> list = Arrays.asList(values);
recipe.setUserCategories(list);
还有一个 RecipeBO:
Recipe recipe = new Recipe();
List<RecipeUserCategoryRel> recipes = new ArrayList<>();
recipes.add(new RecipeUserCategoryRel("abc", "1"));
recipes.add(new RecipeUserCategoryRel("klm", "1"));
recipes.add(new RecipeUserCategoryRel("xyz", "1"));
recipes.add(new RecipeUserCategoryRel("pqr", "1"));
recipe.setRecipeUserCategoryRels(recipes);
当我将 RecipeBO 映射到 Recipe 时:
Recipe actual = mapper.map(getRecipeBO(), Recipe.class);
我得到下一个输出:
输出:
- RecipeUserCategoryRel(userCategory=abc, userId=1)
- RecipeUserCategoryRel(userCategory=klm, userId=1)
- RecipeUserCategoryRel(userCategory=xyz, userId=1)
- RecipeUserCategoryRel(userCategory=pqr, userId=1)
关于java - ModelMapper:将集合映射到其他结构的集合,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31907218/