在.NET 4.0(当然也包括4.0以前的版本)下,用反射判断某个方法是否运用了自定义Attribute时,可以通过调用MethodInfo的IsDefined()方法进行确认。当然,IsDefined()方法事实上定义在MethodInfo的父类MemberInfo中,但它仅仅被定义为抽象方法,真正的实现是在MethodInfo的子类DynamicMethod中。调用方式如下所示:
复制代码 methodInfo.IsDefined(typeof(MyAttribute), false)
然而,在实际开发中,我发现该方法有一个问题。如果获得MethodInfo的方式是通过加载程序集,然后利用反射方式获得的MethodInfo对象,即使该方法运用了自定义Attribute,返回的结果仍然是false。例如,我们将需要判断的方法所在的类定义到一个单独的Project中,并编译为单独的dll文件,然后,利用Assembly的LoadFile()方式获得程序集:
复制代码 var assembly = Assembly.LoadFile(assemblyPath);
var types = assembly.GetExportedTypes();
types.ToList().ForEach(
type =>
{
var flag =
type.GetMethods().Where(methodInfo => !methodInfo.IsAbstract).Any(
methodInfo => methodInfo.IsDefined(typeof(MyAttribute), false));
Console.WriteLine("Flag of IsDefined is: {0}", flag);
}
);
打印出来的值为false。
反之,如果不是通过加载程序集,而是直接通过typeof()获得的Type,并调用其下MethodInfo.IsDefined()方法,只要该方法被运用了指定的Attribute,返回的结果则为true。
分析原因,大约是获得Type的方式不同所造成的。Assembly类的GetExportedType()实现如下所示:
复制代码 [SecuritySafeCritical]
public override Type[] GetExportedTypes()
{
Type[] o = null;
GetExportedTypes(this.GetNativeHandle(), JitHelpers.GetObjectHandleOnStack<Type[]>(ref o));
return o;
}
注意,这里返回的Type[]事实上是通过引用方式传递给了JitHelpers的GetObjectHandleOnStack<Type[]>方法中:










