C# IsDefined的問題
在.NET 4.0(當然也包括4.0以前的版本)下,用反射判斷某個方法是否運用了自定義Attribute時,可以通過調(diào)用MethodInfo的IsDefined()方法進行確認。當然,IsDefined()方法事實上定義在MethodInfo的父類MemberInfo中,但它僅僅被定義為抽象方法,真正的實現(xiàn)是在MethodInfo的子類DynamicMethod中。調(diào)用方式如下所示:
然而,在實際開發(fā)中,我發(fā)現(xiàn)該方法有一個問題。如果獲得MethodInfo的方式是通過加載程序集,然后利用反射方式獲得的MethodInfo對象,即使該方法運用了自定義Attribute,返回的結(jié)果仍然是false。例如,我們將需要判斷的方法所在的類定義到一個單獨的Project中,并編譯為單獨的dll文件,然后,利用Assembly的LoadFile()方式獲得程序集:
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,并調(diào)用其下MethodInfo.IsDefined()方法,只要該方法被運用了指定的Attribute,返回的結(jié)果則為true。
分析原因,大約是獲得Type的方式不同所造成的。Assembly類的GetExportedType()實現(xiàn)如下所示:
public override Type[] GetExportedTypes()
{
Type[] o = null;
GetExportedTypes(this.GetNativeHandle(), JitHelpers.GetObjectHandleOnStack<Type[]>(ref o));
return o;
}
注意,這里返回的Type[]事實上是通過引用方式傳遞給了JitHelpers的GetObjectHandleOnStack<Type[]>方法中:
internal static ObjectHandleOnStack GetObjectHandleOnStack<T>(ref T o) where T: class
{
TypedReference reference = __makeref(o);
return new ObjectHandleOnStack(reference.GetPointerOnStack());
}
這里將Type轉(zhuǎn)換成了TypedReference。關鍵大約就是這里,可惜我無法找到typeof()的具體實現(xiàn)方式。代碼追蹤到這里,就無法判斷這里發(fā)生的真實原因了。若要了解.NET底層機制的同學,可以告訴我。
若要解決反射方式無法通過IsDefined()判斷的問題,可以調(diào)用MethodInfo的GetCustomAttribute()方法。例如:
{
return methodInfo.GetCustomAttributes(attributeType, false).ToString().Contains(attributeName);
}
無論是利用反射加載,還是使用typeof,采用這種方式判斷方法是否運用了指定的Attribute,都是能夠生效的。
以上就是C#IsDefined的問題的全部內(nèi)容,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Unity報錯InvalidOperationException: out of sync的解決
今天在做個東西,發(fā)現(xiàn)報錯,特此來記錄一下,本文介紹了Unity報錯InvalidOperationException: out of sync的解決,感興趣的可以了解一下2021-05-05C#使用DateTime.Now靜態(tài)屬性動態(tài)獲得系統(tǒng)當前日期和時間
本文主要介紹了C#使用DateTime.Now靜態(tài)屬性動態(tài)獲得系統(tǒng)當前日期和時間,DateTime結(jié)構(gòu)的Now靜態(tài)屬性只是得到一個系統(tǒng)時間對象,該時間對象不會隨著系統(tǒng)時間的變化而變化,如果要動態(tài)顯示系統(tǒng)時間,可以使用計時器間隔地獲取系統(tǒng)時間對象并顯示,感興趣的可以了解一下2024-01-01C# MeasureString測量字符串函數(shù)的使用方法
這篇文章主要介紹了C# MeasureString測量字符串函數(shù)的使用方法,需要的朋友可以參考下2014-10-10