C#使用反射进行类型比较

时间:2011-12-09 22:08:50

标签: c# reflection comparison

我想使用反射来检查属性是否为DbSet<T>类型。

public class Foo
{
    public DbSet<Bar> Bars { get; set; }
}

使用反射:

var types = Assembly.GetExecutingAssembly().GetTypes();
foreach (var type in types)
{
    if (type.IsSubclassOf(typeof (Foo)) || type.FullName == typeof (Foo).FullName)
    {
        foreach (
            var prop in Type.GetType(type.FullName).
                GetProperties(BindingFlags.Public | BindingFlags.DeclaredOnly | BindingFlags.Instance))
        {
            var propType = prop.PropertyType;
            bool a = propType.IsAssignableFrom(typeof (DbSet<>));
            bool b = typeof (DbSet<>).IsAssignableFrom(propType);

            bool c = propType.BaseType.IsAssignableFrom(typeof (DbSet<>));
            bool d = typeof (DbSet<>).IsAssignableFrom(propType.BaseType);


            bool e = typeof (DbSet<>).IsSubclassOf(propType);
            bool f = typeof (DbSet<>).IsSubclassOf(propType.BaseType);
            bool g = propType.IsSubclassOf(typeof (DbSet<>));
            bool h = propType.BaseType.IsSubclassOf(typeof (DbSet<>));

            bool i = propType.BaseType.Equals(typeof (DbSet<>));
            bool j = typeof (DbSet<>).Equals(propType.BaseType);

            bool k = propType.Name == typeof (DbSet<>).Name;
        }
    }
}
  • 是否有合并的解决方案来检查类型?如您所见,我正在使用IsSubClassOf + FullName来获取类型Foo的类以及从Foo派生的任何其他类。

  • 除了c,f,k之外的所有检查(a到j)都返回false。 c,f将System.Object作为BaseType返回,对我来说没用。 k,我认为不安全检查。但如果找不到其他解决方法,那将是我使用的。在调试模式下,propType的{​​{1}}为:

     
    System.Data.Entity.DbSet`1[[Console1.Bar, ConsoleApplication1, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null]]

    还有其他方法可以检查FullName是否属于propType类型吗? 感谢。

3 个答案:

答案 0 :(得分:5)

您是否还需要它来处理DbSet<>的子类?如果没有,您可以使用:

if (propType.IsGenericType &&
    propType.GetGenericTypeDefinition() == typeof(DbSet<>))

完整样本:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;

class Test<T>
{
    public List<int> ListInt32 { get; set; }
    public List<T> ListT { get; set; }
    public string Other { get; set; }
    public Action<string> OtherGeneric { get; set; }
}

class Program
{
    static void Main(string[] args)
    {
        var query = from prop in typeof(Test<string>).GetProperties()
                    let propType = prop.PropertyType
                    where propType.IsGenericType &&
                          propType.GetGenericTypeDefinition() == typeof(List<>)
                    select prop.Name;

        foreach (string name in query)
        {
            Console.WriteLine(name);
        }
    }
}

使用子类,您只需要在继承层次结构中递归地应用相同的测试。如果你需要测试接口,那就更难了。

答案 1 :(得分:2)

您的代码存在的问题是假设typeof(DbType<>)表示常规类型。它不是普通类型:相反,它是泛型类型定义。这就是IsAssignableFromIsSubclassOf等不起作用的原因。如果满足以下条件,则Type x的类型为DbSet<T>

x.IsGenericType && x.GetGenericTypeDefinition() == typeof(DbSet<>) && x.GetGenericTypeArguments()[0] == typeof(T)

答案 2 :(得分:0)

我建议进行以下类型比较(对我来说效果很好):

foreach (var prop in customObject.GetType().GetProperties().Where(e => e.CanRead && e.CanWrite))
{
    var typeName = prop.PropertyType.FullName ?? "";
    //!!! Type comparision here:
    if (typeName == typeof(string).FullName)
    {
        prop.SetValue(customObject, "abc");
        continue;
    }
    // other code
}

对于可为空的类型(例如 DateTime? ),您可以使用这种类型的比较:

if (typeName.Contains(typeof(string).FullName))
{
    prop.SetValue(customObject, "abc");
}