使用反射检查IEnumerable <t> </t>

时间:2011-04-06 13:59:48

标签: c# reflection valueinjecter

编辑

这个问题的简单版本是,如果我有一些object o,我将如何检查是否某种类型实现IEnumerable<string> 带反射?最初的问题更具体,但对上述问题的答案也同样好。很抱歉,如果我在这个问题上提供太多的详细信息

结束编辑

以下是一个人为的ValueInjecter POC。除了最底层的isCollectionMapping方法之外,一切都很顺利。当且仅当源属性和目标属性都是实现IEnumerable<respectiveTypes>的任何对象时,我试图让它返回true。

我已尝试IsAssignableFromIsInstanceOfType,但似乎都没有效果。

其他一切都有效,因为当我取消注释方法的第二行以明确检查名称“Children”的属性时,它可以正常工作。

注意 - 我知道此示例存在问题。也就是说,我正在尝试检查任何旧IEnumerable<>,但始终知道返回List<>;在这一点上,它只是一个愚蠢的概念证明。

[TestClass]
public class UnitTest1 {

    [TestMethod]
    public void TestMethod1() {
        List<string> strings = new List<string>();

        Subject S = new Subject() {
            id = 1,
            SubjectName = "S1",
            Children = { new Subject() { id = 2, SubjectName = "S1a" },
                         new Subject() { id = 3, SubjectName = "S1b", Children = { new Subject() { id = 4} } } }
        };

        SubjectViewModel VM = (SubjectViewModel)new SubjectViewModel().InjectFrom<CollectionToCollection>(S); ;


        Assert.AreEqual(2, VM.Children.Count);
        Assert.AreEqual(1, VM.Children.Single(s => s.id == 3).Children.Count);
    }
}


public class Subject {
    public Subject() {
        Children = new List<Subject>();
    }

    public string SubjectName { get; set; }
    public int id { get; set; }

    public List<Subject> Children { get; set; }
}

public class SubjectViewModel {
    public SubjectViewModel() {
        Children = new List<SubjectViewModel>();
    }

    public string SubjectName { get; set; }
    public int id { get; set; }

    public List<SubjectViewModel> Children { get; set; }
}

public class CollectionToCollection : Omu.ValueInjecter.ConventionInjection {
    protected override bool Match(ConventionInfo c) {
        return c.TargetProp.Name == c.SourceProp.Name;
    }

    protected override object SetValue(ConventionInfo c) {
        if (isCollectionMapping(c))
            return (c.SourceProp.Value as IEnumerable<Subject>).Select(s => (SubjectViewModel)(new SubjectViewModel().InjectFrom<CollectionToCollection>(s))).ToList();
        else
            return c.SourceProp.Value;
    }

    private bool isCollectionMapping(ConventionInfo c) {
        return c.SourceProp.Value.GetType().IsInstanceOfType(typeof(IEnumerable<Subject>)) && c.TargetProp.Value.GetType().IsAssignableFrom(typeof(IEnumerable<SubjectViewModel>));

        //return c.SourceProp.Name == "Children" && c.TargetProp.Name == "Children";
    }
}

3 个答案:

答案 0 :(得分:29)

  

如果我有一些对象o,我该怎么做?   检查o是否属于某种类型   实现IEnumerable<string>

简单如下:

o is IEnumerable<string>

顺便说一句,你当前的代码没有工作,因为它正在逆转对可分配性关系的测试(就好像这个方法被称为IsAssignableTo),即假设:

Bar bar = ...
Foo foo = bar

暗示:

typeof(Bar).IsAssignableFrom(typeof(Foo)) // wrong

实际上,实际意义是:

typeof(Foo).IsAssignableFrom(typeof(Bar))
  

即,我正在尝试检查任何内容   旧IEnumerable<>

在这种情况下,您需要测试类型是否实现了通用接口的构造版本:

o.GetType()
 .GetInterfaces()
 .Any(t => t.IsGenericType 
        && t.GetGenericTypeDefinition() == typeof(IEnumerable<>))

答案 1 :(得分:10)

  

这个问题的简单版本是,如果我有一些对象o,我将如何检查o是否是某种类型实现IEnumerable<string>

像这样:

object o = whatever;
bool isSequenceOfStrings = o is IEnumerable<string>;

答案 2 :(得分:5)

我本可以错过一些东西(没有读过你的整个代码示例),但这似乎不需要在这里反思。

如何使用:

if (c.SourceProp.Value is IEnumerable<Subject>)
   return true;

如果您不知道具体类型,请使用泛型:

public bool MyFunction<T>(...)
{
   if (c.SourceProp.Value is IEnumerable<T>)
      return true;
}

或者如果您需要使用界面,请这样做(保存演员表):

var enumerable = c.SourceProp.Value as IEnumerable<string>;
if (enumerable != null)
{
   // Use IEnumerable<string>
   return true;
}

return false;