在.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[]>方法中:
[TargetedPatchingOptOut("Performance critical to inline across NGen image boundaries"), SecurityCritical]
internal static ObjectHandleOnStack GetObjectHandleOnStack<T>(ref T o) where T: class
{
TypedReference reference = __makeref(o);
return new ObjectHandleOnStack(reference.GetPointerOnStack());
}
這里將Type轉換成了TypedReference。關鍵大約就是這里,可惜我無法找到typeof()的具體實現方式。代碼追蹤到這里,就無法判斷這里發生的真實原因了。若要了解.NET底層機制的同學,可以告訴我。
若要解決反射方式無法通過IsDefined()判斷的問題,可以調用MethodInfo的GetCustomAttribute()方法。例如:
private static bool IsAppliedWith(this MethodInfo methodInfo, Type attributeType, string attributeName)
{
return methodInfo.GetCustomAttributes(attributeType, false).ToString().Contains(attributeName);
}
無論是利用反射加載,還是使用typeof,采用這種方式判斷方法是否運用了指定的Attribute,都是能夠生效的。
以上就是C#IsDefined的問題的全部內容,希望能給大家一個參考,也希望大家多多支持武林網。