是否有一种简单的方法来合并C#匿名对象

ajm*_*jma 28 c# anonymous-objects

假设我有两个这样的匿名对象:

var objA = new { test = "test", blah = "blah" };
var objB = new { foo = "foo", bar = "bar" };
Run Code Online (Sandbox Code Playgroud)

我想将它们结合起来得到:

new { test = "test", blah = "blah", foo = "foo", bar = "bar" };
Run Code Online (Sandbox Code Playgroud)

我不知道在编译时objA和objB的属性是什么.我希望这就像jquery的extend方法.

有人知道可以帮我这样做的库或.net框架类吗?

Luk*_*ust 26

如果您真的在C#4.0意义上表示动态,那么您可以执行以下操作:

static dynamic Combine(dynamic item1, dynamic item2)
{
    var dictionary1 = (IDictionary<string, object>)item1;
    var dictionary2 = (IDictionary<string, object>)item2;
    var result = new ExpandoObject();
    var d = result as IDictionary<string, object>; //work with the Expando as a Dictionary

    foreach (var pair in dictionary1.Concat(dictionary2))
    {
        d[pair.Key] = pair.Value;
    }

    return result;
}
Run Code Online (Sandbox Code Playgroud)

您甚至可以使用带有两个对象(不是动态的)的反射来编写一个版本并返回一个动态.

  • 这只适用于两个动态项都实现了对`IDictionary <string,object>`的显式转换 - 这对于`ExpandoObject`是正确的,但对于强制转换为`dynamic`的匿名类型则不然(这就是它为@JoshRusso崩溃的原因) ) (5认同)
  • 你的代码中有一个错误 - 你应该将item1和item2转换为IDictionary,而不是Dictionary.所以不想让我编辑,因为它少于6个字符...... (3认同)
  • 这对我不起作用。我收到以下错误:无法将类型'&lt;&gt; f__AnonymousType0`1 [System.Int32]'的对象强制转换为类型'System.Collections.Generic.IDictionary`2 [System.String,System.Object]。有什么建议么? (2认同)

Yve*_* M. 6

使用TypeDescriptor

正如评论中所提到的,Luke Foust解决方案不适用于匿名类型.强制转换为Dictionary不起作用,我建议使用TypeDescriptor.GetProperties方法构造字典:

public static dynamic CombineDynamics(object object1, object object2)
{
  IDictionary<string, object> dictionary1 = GetKeyValueMap(object1);
  IDictionary<string, object> dictionary2 = GetKeyValueMap(object2);

  var result = new ExpandoObject();

  var d = result as IDictionary<string, object>;
  foreach (var pair in dictionary1.Concat(dictionary2))
  {
    d[pair.Key] = pair.Value;
  }

  return result;
}

private static IDictionary<string, object> GetKeyValueMap(object values)
{
  if (values == null)
  {
    return new Dictionary<string, object>();
  }

  var map = values as IDictionary<string, object>;
  if (map == null)
  {
    return map;
  }

  map = new Dictionary<string, object>();
  foreach (PropertyDescriptor descriptor in TypeDescriptor.GetProperties(values))
  {
    map.Add(descriptor.Name, descriptor.GetValue(values));
  }

  return map;
}
Run Code Online (Sandbox Code Playgroud)

它现在使用匿名类型:

var a = new {foo = "foo"};
var b = new {bar = "bar"};

var c = CombineDynamics(a, b);

string foobar = c.foo + c.bar;
Run Code Online (Sandbox Code Playgroud)

笔记:

我的GetKeyValueMap方法基于RouteValueDictionary类(System.Web.Routing).我已经重写了它(使用ILSpy来反汇编它),因为我认为一个System.Web类与对象合并无关.

  • 它应该是`if(map!= null)`,否则它将不适用于匿名类型. (5认同)