AutoMapper,将集合的嵌套集合映射到列表

AutoMapper, Mapping a nested collection of collections to List

我刚刚开始在一个 asp 网络核心项目上使用 AutoMapper,我正在尝试将一个包含一个对象集合的对象映射到一个实体。

来源

public class MyClass
    {
        public List<MyCollection> MyCollections { get; set; }
    }

    public class MyCollection
    {
        public int CollectionId { get; set; }
        public List<Description> Descriptions { get; set; }
    }

    public class Description
    {
        public int DescriptionId { get; set; }
        public string Text { get; set; }
    }

目的地

 public class DescriptionToCollection
    {
        public int DescriptionId { get; set; }
        public int CollectionId { get; set; }
    }

我试过 ConvertUsing 的想法是这样的,但我无法让它工作。

CreateMap<MyClass, List<DescriptionToCollection>>()
             .ConvertUsing(source => source.MyCollections.Select(x =>x.Description.Select(y=> new DescriptionToCollection{ DescriptionId=y.DescriptionId,CollectionId=x.CollectionId}).ToList()
             ));

搜索 AutoMappers 文档和互联网,我找不到与我的问题类似的内容。 非常感谢任何帮助。

除了示例代码中的拼写错误外,您几乎已经成功了。因为您没有在顶层映射 1:1,所以您需要在某处展平,然后使用 SelectMany 执行此操作,适当移动 ToList 调用。

CreateMap<MyClass, List<DescriptionToCollection>>()
    .ConvertUsing(source => source.MyCollections.SelectMany(x => // SelectMany to flatten
                        x.Descriptions.Select(y => 
                            new DescriptionToCollection
                            {
                                DescriptionId = y.DescriptionId,
                                CollectionId = x.CollectionId
                            }
                        ) // ToList used to be here
                    ).ToList()
                );

您没有理解自动映射器的用途。

它应该用于将一种类型的输入对象转换为另一种类型的输出对象。

并且您想创建一个从 MyClass 类型到 List 的映射 - 这应该被视为投影。

您可以使用 LINQ 实现(例如作为 MyClass 的扩展方法):

    public static class MyClassExtension
    {
        public static List<DescriptionToCollection> ToDescriptionToCollection(this MyClass value)
        {
            return value.MyCollections.SelectMany(mc => mc.Descriptions.Select(d => new DescriptionToCollection()
            {
                CollectionId = mc.CollectionId,
                DescriptionId = d.DescriptionId
            })).ToList();
        }
    }

尝试实现ITypeConverter,按照示例代码:

你的类

    public class Class1
    {
        public List<Class2> class2 { get; set; }
    }

    public class Class2
    {
        public int CollectionId { get; set; }
        public List<Class3> class3 { get; set; }
    }

    public class Class3
    {
        public int DescriptionId { get; set; }
        public string Text { get; set; }
    }


    public class ClassDto
    {
        public int DescriptionId { get; set; }
        public int CollectionId { get; set; }
    }

自定义地图

    public class ClassCustomMap : ITypeConverter<Class1, List<ClassDto>>
    {

        public List<ClassDto> Convert(Class1 source, List<ClassDto> destination, ResolutionContext context)
        {
            var classDtoList = new List<ClassDto>();
            foreach (var item in source.class2)
            {
                var collectionID = item.CollectionId;
                foreach (var obj in item.class3)
                {
                    var classDto = new ClassDto();
                    classDto.CollectionId = collectionID;
                    classDto.DescriptionId = obj.DescriptionId;

                    classDtoList.Add(classDto);
                }
            }

            return classDtoList;
        }
    }

映射声明

    CreateMap<Class1, List<ClassDto>>().ConvertUsing<ClassCustomMap>();

使用方法

    var class2 = new Class2();
    class2.CollectionId = 2;

    var class3 = new Class3();
    class3.DescriptionId = 1;
    class3.Text = "test";

    class2.class3 = new System.Collections.Generic.List<Class3>() { class3 };

    var class1 = new Class1();
    class1.class2 = new System.Collections.Generic.List<Class2>() { class2 };

    var result = mapper.Map<List<ClassDto>>(class1);

为了清晰和简化,我使用了显式循环,如果您愿意,可以使用 LinQ 和 Lambda 优化转换函数