在新的.NET Framework 4.5中使用反射时,我遇到了一个奇怪的行为,我发现这是非常意外的。命名空间System.Reflection为利用Type对象提供了一些新的扩展方法。其中两个是GetRuntimeProperty(字符串名称)和GetRuntimeProperties()。
现在假设你有一个带有内部属性的简单对象。
public class ObjectBase
{
protected int Id { get; set; }
public string Name { get; set; }
}
现在你尝试利用这种类型。
var properties = typeof(ObjectBase).GetRuntimeProperties();
// properties.Count = 2
var idProperty = typeof(ObjectBase).GetRuntimeProperty("Id");
var nameProperty = typeof(ObjectBase).GetRuntimeProperty("Name");
// idProperty = null
// nameProperty = System.String Name
正如预期的那样,properties
对象包含Id和Name属性定义的两个属性定义,而nameProperty保存Name属性定义。不期望的是idProperty
对象为null ...
来自.NET Framework,我想这是微软架构师的意图,但我必须说这似乎不是你真正希望发生的事情。我相信这些类似的方法应该表现相同但似乎GetRuntimeProperty过滤了GetRuntimeProperties不应用过滤器的公共属性。
有没有人对微软为什么决定这些类似的方法应该有不同的行为有合理的解释?设计错误?
感谢。
答案 0 :(得分:3)
内部GetRuntimeProperty
来电Type.GetProperty(name)
,用于搜索具有指定名称的公共属性。属性Id
受到保护,因此无法找到。
public static PropertyInfo GetRuntimeProperty(this Type type, string name)
{
CheckAndThrow(type);
return type.GetProperty(name);
}
另一方面,GetRuntimeProperties
返回公共和非公共属性
public static IEnumerable<PropertyInfo> GetRuntimeProperties(this Type type)
{
CheckAndThrow(type);
return type.GetProperties(BindingFlags.NonPublic | BindingFlags.Public |
BindingFlags.Static | BindingFlags.Instance);
}
Explanation: GetRuntimeProperties
目的是返回所有属性的IEnumerable<PropertyInfo>
集合,让您通过 LINQ 过滤该集合。您可以选择公共,非公共或任何其他类型的属性。由GetRuntimeProperty
返回的单一属性,您不需要这种灵活性,因此它在大多数常见用途中受到限制。