динамическое приведение типов в пользовательской инъекции ValueInjecter - PullRequest
3 голосов
/ 02 мая 2011

Я создал пользовательский класс внедрения для ValueInjecter , который выполняет рекурсивное внедрение для общих дочерних коллекций, но работает с существующими целевыми объектами, а не клонирует, как образец "CloneInjection" на сайте VI. Однако в настоящее время я выполняю приведение к известному типу (ICollection ), поэтому класс инъекции подходит только для одного типа с жестким кодом. Кажется, я не могу придумать способ динамически разыграть универсальный тип, какие-либо предложения? Вот код SetValue для моего класса "RecursiveInjection".

//for value types and string just return the value as is
if (c.SourceProp.Type.IsValueType || c.SourceProp.Type == typeof(string))
  return c.SourceProp.Value;

if (c.SourceProp.Type.IsGenericType)
  {
  //handle IEnumerable<> also ICollection<> IList<> List<>
  if (c.SourceProp.Type.GetGenericTypeDefinition().GetInterfaces().Contains(typeof(IEnumerable)))
    {
      var t = c.TargetProp.Type.GetGenericArguments()[0];
      if (t.IsValueType || t == typeof(string)) return c.SourceProp.Value;

      //get enumerable object in target
     var targetCollection = c.TargetProp.Value as ICollection<MyTargetType>;
     //possible to cast dynamically?

     foreach (var o in c.SourceProp.Value as IEnumerable)
     {
       //get ID of source object
       var sourceID = (int)o.GetProps().GetByName("ID").GetValue(o);
       //find matching target object if there is one
       var target = targetCollection.SingleOrDefault(x => x.ID == sourceID);
       if (target != null)
       {
         target.InjectFrom<RecursiveInjection>(o);
       }
     }
    return targetCollection;
  }
}

Спасибо, Дано

Ответы [ 3 ]

1 голос
/ 05 мая 2011

Хотя я не смог придумать способ динамического приведения универсальной коллекции, я обнаружил решение основной проблемы, которую я пытался решить, которая заключалась в том, чтобы внедрить объекты EF POCO, содержащие дочерние коллекции (одно- многие "свойства навигации").

Я только что обновился с EF 4.0 до EF 4.1, и аллилуйя! EF 4.1 фактически управляет объединением объектов в дочерних коллекциях. После обновления я могу использовать CloneInjection как есть.

1 голос
/ 03 мая 2011

Я бы просто использовал ICollection и переопределил Equals (объект) для ваших сравнений.

public class MyTargetType
{
    public override bool Equals( object obj )
    {
        return ( obj is MyTargetType )
                    ? this.ID == ( ( MyTargetType ) obj ).ID
                    : false;
    }
}

Тогда в вашем цикле вы бы сопоставили объект напрямую, а не использовали отражение.

var target = targetCollection.SingleOrDefault( x => x.Equals( o ) );
0 голосов
/ 07 мая 2011

Я думаю, это то, что вы ищете:

    public class M1
    {
        public string Name { get; set; }
    }

    public class M2
    {
        public string Name { get; set; }
    }

    [Test]
    public void Cast()
    {
        object source = new List<M1> {new M1 {Name = "o"}};
        object target = new List<M2>();


        var targetArgumentType = target.GetType().GetGenericArguments()[0];

        var list = Activator.CreateInstance(typeof(List<>).MakeGenericType(targetArgumentType));
        var add = list.GetType().GetMethod("Add");

        foreach (var o in source as IEnumerable)
        {
            var t = Activator.CreateInstance(targetArgumentType);
            add.Invoke(list, new[] { t.InjectFrom(o) });
        }

        target = list;

        Assert.AreEqual("o", (target as List<M2>).First().Name);
    }

с .NET 4 вы можете сделать это немного короче, используя динамический

...