2

更新:这些东西已经演变成一个不错的项目,请参见http://valueinjecter.codeplex.com


看看这个,我刚刚写了一个简单的自动映射器,它从一个对象的具有相同名称和类型的属性中获取值并将其放入另一个对象中,您可以为您可能需要的每种类型添加异常(ifs、switch)

所以告诉我你怎么看?

我这样做了,所以我可以做这样的事情:

Product –> ProductDTO

ProductDTO –> Product

这就是它的开始:

我在我的 Inputs/Dto/ViewModels 中为 DropDowns 使用“对象”类型,因为我向 html 发送了 IEnumerable<SelectListItem> 并且我收到了返回的选定键的字符串数组

 public void Map(object a, object b)
    {
        var pp = a.GetType().GetProperties();
        foreach (var pa in pp)
        {
            var value = pa.GetValue(a, null);

            // property with the same name in b
            var pb = b.GetType().GetProperty(pa.Name);
            if (pb == null)
            {
                //no such property in b
                continue;
            }

            if (pa.PropertyType == pb.PropertyType)
            {
                pb.SetValue(b, value, null);
            }

        }
    }

更新: 真正的用法:
构建方法(输入= Dto):

        public static TI BuildInput<TI, T>(this T entity) where TI: class, new()
        {
            var input = new TI();
            input = Map(entity, input) as TI;
            return input;
        }

        public static T BuildEntity<T, TI, TR>(this TI input)
            where T : class, new()
            where TR : IBaseAdvanceService<T>
        {               
            var id = (long)input.GetType().GetProperty("Id").GetValue(input, null);
            var entity = LocatorConfigurator.Resolve<TR>().Get(id) ?? new T();
            entity = Map(input, entity) as T;
            return entity;
        }

        public static TI RebuildInput<T, TI, TR>(this TI input)
            where T: class, new()
            where TR : IBaseAdvanceService<T>
            where TI : class, new()
        {

                return input.BuildEntity<T, TI, TR>().BuildInput<TI, T>();
            }

在控制器中:

    public ActionResult Create()
    { 
        return View(new Organisation().BuildInput<OrganisationInput, Organisation>());
    }

    [AcceptVerbs(HttpVerbs.Post)]
    public ActionResult Create(OrganisationInput o)
    {
        if (!ModelState.IsValid)
        {
            return View(o.RebuildInput<Organisation,OrganisationInput, IOrganisationService>());                
        }
        organisationService.SaveOrUpdate(o.BuildEntity<Organisation, OrganisationInput, IOrganisationService>());
        return RedirectToAction("Index");
    }

真正的 Map 方法

public static object Map(object a, object b)
        {
            var lookups = GetLookups();

            var propertyInfos = a.GetType().GetProperties();
            foreach (var pa in propertyInfos)
            {
                var value = pa.GetValue(a, null);

                // property with the same name in b
                var pb = b.GetType().GetProperty(pa.Name);
                if (pb == null)
                {
                    continue;
                }

                if (pa.PropertyType == pb.PropertyType)
                {
                    pb.SetValue(b, value, null);
                }
                else if (lookups.Contains(pa.Name) && pa.PropertyType == typeof(LookupItem))
                {
                    pb.SetValue(b, (pa.GetValue(a, null) as LookupItem).GetSelectList(pa.Name), null);
                }
                else if (lookups.Contains(pa.Name) && pa.PropertyType == typeof(object))
                {
                    pb.SetValue(b, pa.GetValue(a, null).ReadSelectItemValue(), null);
                }
                else if (pa.PropertyType == typeof(long) && pb.PropertyType == typeof(Organisation))
                {
                    pb.SetValue(b, pa.GetValue<long>(a).ReadOrganisationId(), null);
                }
                else if (pa.PropertyType == typeof(Organisation) && pb.PropertyType == typeof(long))
                {
                    pb.SetValue(b, pa.GetValue<Organisation>(a).Id, null);
                }
            }

            return b;
        }
4

4 回答 4

6

您可能想要添加的一件事是缓存反射位。如果您映射一个对象两次,您可能不想再次查找所有反射的东西。此外,像 GetValue 和 SetValue 这样的东西很慢,我改用后期绑定的委托 + Reflection.Emit 来加快速度。

于 2009-11-23T13:36:37.837 回答
6

只需使用AutoMapper。这很好,但它会成长为一个迷你项目。

AM(真正的)做的一些事情是:

  • 报告您是否有无法映射到的属性
  • 展平物体
  • 提供钩子供你自定义某些方面,而不是在一个大的 switch 语句中
  • 出于性能原因而不是直接反射使用 Expression.Compile

但这肯定是一个有趣的空间,自动映射的想法当然很有用。

有点像1533行中的 DI vs NInject 或其朋友 - 很酷,但为什么呢?

我认为您已阅读Jimmy 博客上有关 2 路映射的文章和评论

于 2009-11-23T08:08:08.920 回答
3

只是一个想法:

如果抽象如此容易地映射到被抽象的对象,人们可能想知道抽象的意义何在。

于 2009-11-23T08:30:06.043 回答
3

您可以为可能需要的每种类型添加例外(ifs、switch)

你不会这样做的。严重地。作用于对象类型的 case 语句是糟糕的 OOP 风格。我的意思是,真的很糟糕。就像在肚子里喝着伏特加酒开车一样。它可能会工作一段时间,但最终你会遇到麻烦。

好的,吉米告诉你不要使用 AutoMapper ......但我敢打赌他的意思是别的。现在,你有没有发明一些不同的东西——让吉米开心的东西?;-) 不,您刚刚制作了自己的半卷 AutoMapper。吉米告诉你不要使用它!;-)

所以这是我的建议:忽略 Jimmy 所说的话,想想你自己.. 并使用 AutoMapper ;-)

于 2009-11-23T09:58:39.770 回答