C# IsDefined的問題
在.NET 4.0(當(dāng)然也包括4.0以前的版本)下,用反射判斷某個(gè)方法是否運(yùn)用了自定義Attribute時(shí),可以通過調(diào)用MethodInfo的IsDefined()方法進(jìn)行確認(rèn)。當(dāng)然,IsDefined()方法事實(shí)上定義在MethodInfo的父類MemberInfo中,但它僅僅被定義為抽象方法,真正的實(shí)現(xiàn)是在MethodInfo的子類DynamicMethod中。調(diào)用方式如下所示:
然而,在實(shí)際開發(fā)中,我發(fā)現(xiàn)該方法有一個(gè)問題。如果獲得MethodInfo的方式是通過加載程序集,然后利用反射方式獲得的MethodInfo對(duì)象,即使該方法運(yùn)用了自定義Attribute,返回的結(jié)果仍然是false。例如,我們將需要判斷的方法所在的類定義到一個(gè)單獨(dú)的Project中,并編譯為單獨(dú)的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()方法,只要該方法被運(yùn)用了指定的Attribute,返回的結(jié)果則為true。
分析原因,大約是獲得Type的方式不同所造成的。Assembly類的GetExportedType()實(shí)現(xiàn)如下所示:
public override Type[] GetExportedTypes()
{
Type[] o = null;
GetExportedTypes(this.GetNativeHandle(), JitHelpers.GetObjectHandleOnStack<Type[]>(ref o));
return o;
}
注意,這里返回的Type[]事實(shí)上是通過引用方式傳遞給了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。關(guān)鍵大約就是這里,可惜我無法找到typeof()的具體實(shí)現(xiàn)方式。代碼追蹤到這里,就無法判斷這里發(fā)生的真實(shí)原因了。若要了解.NET底層機(jī)制的同學(xué),可以告訴我。
若要解決反射方式無法通過IsDefined()判斷的問題,可以調(diào)用MethodInfo的GetCustomAttribute()方法。例如:
{
return methodInfo.GetCustomAttributes(attributeType, false).ToString().Contains(attributeName);
}
無論是利用反射加載,還是使用typeof,采用這種方式判斷方法是否運(yùn)用了指定的Attribute,都是能夠生效的。
以上就是C#IsDefined的問題的全部?jī)?nèi)容,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- 簡(jiǎn)單記錄C# 條件編譯
- C#.NET學(xué)習(xí)筆記5 C#中的條件編譯
- 淺談#ifndef,#define,#endif的作用和用法
- 詳解C語言中的#define宏定義命令用法
- typedef和#define的用法以及區(qū)別
- C語言中的內(nèi)聯(lián)函數(shù)(inline)與宏定義(#define)詳細(xì)解析
- C語言中#define與typedef的互換細(xì)節(jié)詳解
- C語言編程技巧 關(guān)于const和#define的區(qū)別心得
- c++ 盡量不要使用#define 而是用const、enum、inline替換。
- C# #define條件編譯詳解
相關(guān)文章
Unity報(bào)錯(cuò)InvalidOperationException: out of sync的解決
今天在做個(gè)東西,發(fā)現(xiàn)報(bào)錯(cuò),特此來記錄一下,本文介紹了Unity報(bào)錯(cuò)InvalidOperationException: out of sync的解決,感興趣的可以了解一下2021-05-05C#使用DateTime.Now靜態(tài)屬性動(dòng)態(tài)獲得系統(tǒng)當(dāng)前日期和時(shí)間
本文主要介紹了C#使用DateTime.Now靜態(tài)屬性動(dòng)態(tài)獲得系統(tǒng)當(dāng)前日期和時(shí)間,DateTime結(jié)構(gòu)的Now靜態(tài)屬性只是得到一個(gè)系統(tǒng)時(shí)間對(duì)象,該時(shí)間對(duì)象不會(huì)隨著系統(tǒng)時(shí)間的變化而變化,如果要?jiǎng)討B(tài)顯示系統(tǒng)時(shí)間,可以使用計(jì)時(shí)器間隔地獲取系統(tǒng)時(shí)間對(duì)象并顯示,感興趣的可以了解一下2024-01-01c# 獲取網(wǎng)頁(yè)中指定的字符串信息的實(shí)例代碼
c# 獲取網(wǎng)頁(yè)中指定的字符串信息的實(shí)例代碼,需要的朋友可以參考一下2013-04-04C# MeasureString測(cè)量字符串函數(shù)的使用方法
這篇文章主要介紹了C# MeasureString測(cè)量字符串函數(shù)的使用方法,需要的朋友可以參考下2014-10-10