4

我将业务对象映射到实体,并且在某些情况下实体的结构与业务对象不同。
我将userCategories其作为字符串存储在业务对象RecipeBo中,因为 BO 不必了解实体的内部结构。这些字符串需要映射到 和 的关系RecipeRecipeUserCategoryRel除此之外,另一个字段userIdRecipeBo需要映射到其中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字段。

4

1 回答 1

3

问题

这是一个复杂的情况,因为您从其他层次结构而不是直接从列表中获取 userId。ModelMapper 会将 List 映射到 List 但如果您不将 ModelMapper 配置为 LOOSE,它将无法工作。

modelMapper.getConfiguration()
  .setMatchingStrategy(MatchingStrategies.LOOSE);

无论如何,如果您以这种方式(松散模式)配置 ModelMapper,它将映射 List 并放入您的 Class RecipeUserCategoryRel 的 String 属性(在这种情况下,例如 userCategory 如果它是一个字符串并且考虑到 userId 不是一个字符串)其他(我不太确定)我认为它会是空的。

解决方案

好吧,我认为您的问题的解决方案是创建一个 Converter 并将其添加到您的 ModelMapper 实例中:

  • RecipeBO(源)-> 配方(目标)

代码如下:

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);

还有一个食谱:

 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)
于 2016-07-28T21:05:09.363 回答