web-dev-qa-db-ja.com

C#匿名オブジェクトをマージする簡単な方法はありますか

このような2つの匿名オブジェクトがあるとしましょう:

var objA = new { test = "test", blah = "blah" };
var objB = new { foo = "foo", bar = "bar" };

それらを組み合わせて次のようにします。

new { test = "test", blah = "blah", foo = "foo", bar = "bar" };

コンパイル時にobjAとobjBの両方のプロパティが何であるかわかりません。これをjqueryのextendメソッドのようにしたいと思います。

これを行うのに役立つライブラリまたは.netフレームワーククラスを知っている人はいますか?

28
ajma

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;
}

2つのオブジェクト(動的ではない)を取り、動的を返すリフレクションを使用してバージョンを作成することもできます。

25
Luke Foust

TypeDescriptorの使用

コメントで述べたように、 Luke Foustソリューション は匿名タイプでは機能しません。辞書へのキャストが機能しないので、 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;
}

現在、匿名タイプで機能しています。

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

var c = CombineDynamics(a, b);

string foobar = c.foo + c.bar;

ノート:

私のGetKeyValueMapメソッドは RouteValueDictionary クラス(System.Web.Routing)に基づいています。 System.Webクラスはオブジェクトのマージとは関係がないと思うので、( ILSpy を使用して逆アセンブルします)書き換えました。

6
Yves M.