映射元素而不创建重复项

本文关键字:创建 元素 映射 | 更新日期: 2023-09-27 18:26:52

我有两个类:

public class Element
{
    public Item Item { get; set; }
}
public class Item
{
    public Element Element { get; set; }
}

对于这些类,我有相同结构的DTO。此方法创建映射的源数据:

static Element[] CreateElements()
{
    var element2 = new Element();
    return new[]
    {
        new Element(new Item(element2)),
        element2,
        new Element()
    };
}

然后我配置映射和映射元素:

Mapper.CreateMap<Element, ElementDto>();
Mapper.CreateMap<Item, ItemDto>();
var elements = CreateElements();
var mappedElements = elements
    .Select(_ => Mapper.Map(_, typeof(Element), typeof(ElementDto)))
    .OfType<ElementDto>()
    .ToArray();

在我检查映射结果后:

foreach (var element in mappedElements)
{
    Console.WriteLine(mappedElements.Any(e => e?.Item?.Element == element));
}

此代码显示三次"False"。因此,"CreateElements"中的"element2"创建了两个副本。

对源元素的相同测试将返回"假-真-假":

foreach (var element in elements)
{
    Console.WriteLine(elements.Any(e => e?.Item?.Element == element));
}

因为我需要配置映射,以免重复元素?有可能吗?

映射元素而不创建重复项

我不认为这是AutoMapper的问题。

您正在创建三个不同的Element项目,并将它们映射到某种类型的ElementDto。它们是三个不同的对象(在结构和引用方面),如果将它们映射到同一类型,则不能指望它们是相等的。

如果你考虑你的物品:

    var element2 = new Element();
    return new[]
    {
        new Element(new Item(element2)),
        element2,
        new Element()
    };

将它们进行比较,你会发现没有一个是平等的。您还没有提供ElementDto类,但我猜您应该实现IEquatable接口,这将确保正确的比较(或重载运算符)。

这可以手动完成。首先,忽略属性Item到AutoMapper没有复制元素链:

Mapper.CreateMap<Item, ItemDto>()
    .ForMember(_ => _.Element, _ => _.Ignore()); 

其次,手动复制带有标记查看项目的链:

static IEnumerable<ElementDto> MapElements(Element[] elements)
{
    var elementToDtoMap = new Dictionary<Element, ElementDto>();
    foreach (var element in elements)
    {
        MapElement(element, null, elementToDtoMap);
    }
    return elementToDtoMap.Select(_ => _.Value);
}
static void MapElement(Element element, ItemDto parentItem, Dictionary<Element, ElementDto> elementToDtoMap)
{
    ElementDto elementDto = null;
    if (elementToDtoMap.TryGetValue(element, out elementDto))
            return;
    elementDto = Mapper.Map<ElementDto>(element);
    elementToDtoMap.Add(element, elementDto);
    if (parentItem != null)
    {
        parentItem.Element = elementDto;
    }
    if (element.Item != null)
    {
        MapElement(element.Item.Element, elementDto.Item, elementToDtoMap);
    }
}