11

我想知道如何映射两个不同对象的字段并将值分配给它。

示例:

public class employee
{
    public int ID { get; set; }
    public string Name { get; set; }
}

public class manager
{
    public int MgrId { get; set; }
    public string MgrName { get; set; }
}

现在我有一个 List 对象。我想将值分配给“经理”类。任何自动的方式来做到这一点。我可以明确地做到这一点并为其赋值。但是我的对象非常大,这就是问题所在。我也不想使用任何第三方工具。

注意:经理不能有任何前缀。它可以是任何东西。(例如:mgrId 可以像 mgrCode)

4

3 回答 3

17

即使忽略属性大小写,您也可以使用反射(注意employee.IDvs. manager.MgrId):

class Program
{
    static void Main(string[] args)
    {
        var employee = new Employee() { ID = 1, Name = "John" };
        var manager = new Manager();
        foreach (PropertyInfo propertyInfo in typeof(Employee).GetProperties())
        {
            typeof(Manager)
                .GetProperty("Mgr" + propertyInfo.Name,
                    BindingFlags.IgnoreCase |
                    BindingFlags.Instance |
                    BindingFlags.Public)
                .SetValue(manager,
                    propertyInfo.GetValue(employee));
        }
    }
}

public class Employee
{
    public int ID { get; set; }
    public string Name { get; set; }
}

public class Manager
{
    public int MgrId { get; set; }
    public string MgrName { get; set; }
}

如果您不知道Mgr前缀,则只能通过后缀进行匹配:

foreach (PropertyInfo propertyInfo in typeof(Employee).GetProperties())
{
    typeof(Manager).GetMembers()
        .OfType<PropertyInfo>()
        .FirstOrDefault(p => p.Name.EndsWith(propertyInfo.Name, 
            StringComparison.CurrentCultureIgnoreCase))
        .SetValue(manager,
            propertyInfo.GetValue(employee));
}

还有一个非常狭隘且不切实际的假设:基于属性顺序的映射(如果您期望这两种类型具有以相同顺序和编号定义的属性,唯一的区别是属性名称)。我不建议任何人在现实生活中使用它,但仍然在这里(只是为了让它更脆弱:)):

typeof(Employee)
    .GetProperties()
    .Select((p, index) =>
        new { Index = index, PropertyInfo = p })
    .ToList()
    .ForEach(p =>
        {
            typeof(Manager)
                .GetProperties()
                .Skip(p.Index)
                .FirstOrDefault()
                .SetValue(manager,
                    p.PropertyInfo.GetValue(employee));
        });
于 2013-07-16T11:45:31.877 回答
13

使用反射或AutoMapper。我推荐后者,因为如果没有目的,编写新代码是一种浪费。

public class Employee
{
    public int Id { get; set; }
    public string Name { get; set; }
}

public class Manager
{
    public int MgrId { get; set; }
    public string MgrName { get; set; }
}

Mapper.Initialize(cfg =>
{
   cfg.RecognizeDestinationPrefixes("Mgr");
   cfg.CreateMap<Employee, Manager>();
});

var manager = Mapper.Map<Employee, Manager>(new Employee { Id = 1, Name = "Fred" });

Console.WriteLine("Id: {0}", manager.MgrId);
Console.WriteLine("Name: {0}", manager.MgrName);

如果属性没有惯用的源标识符,则使用 AutoMapper 的投影

Mapper.CreateMap<Employee, Manager>()
      .ForMember(dest => dest.MgrCode, opt => opt.MapFrom(src => src.ID))
      .ForMember(dest => dest.MgrName, opt => opt.MapFrom(src => src.Name))
于 2013-07-16T11:42:12.317 回答
1

我知道这篇文章现在有点老了,但对于其他正在寻找答案的人来说,这就是你可以追求的……只需使用 lambda 表达式

var managers = employees.Select(x => new Manager()
{
    MgrCode = x.ID,
    MgrName = x.Name
}).ToList();
于 2021-03-03T11:36:24.147 回答