http://blog.sina.com.cn/s/blog_65d6476a0101aa4t.html
2012
僅僅為了獲取函數名,就在函數體中嵌入硬編碼的字符串,這種方法單調乏味還易導致錯誤,不如看一下怎樣使用新的C99特性,在程序運行時(shí)獲取函數名吧?! ?/p> 對象反射庫、調試工具及代碼分析器,經(jīng)常會(huì )需要在運行時(shí)訪(fǎng)問(wèn)函數的名稱(chēng),直到不久前,唯一能完成此項任務(wù)并且可移植的方法,是手工在函數體內嵌入一個(gè)帶有該函數名的硬編碼字符串,不必說(shuō),這種方法非常單調無(wú)奇,并且輕易導致錯誤。本文將要演示怎樣使用新的C99特性,在運行時(shí)獲取函數名?! ?/p> 那么怎樣以編程的方式從當前運行的函數中得到函數名呢?
答案是:使用__FUNCTION__ 及相關(guān)宏。 引出問(wèn)題
通常,在調試中最讓人心煩的階段,是不斷地檢查是否已調用了特定的函數。對此問(wèn)題的解決方法,一般是添加一個(gè)cout或printf()——假如你使用C語(yǔ)言,如下所示:voidmyfunc(){cout<<"myfunc()"<<endl;//其他代碼}
通常在一個(gè)典型的工程中,會(huì )包含有數千個(gè)函數,要在每個(gè)函數中都加入一條這樣的輸出語(yǔ)句,無(wú)疑難過(guò)上“蜀山”啊,因此,需要有一種機制,可以自動(dòng)地完成這項操作?! ?/p>
獲取函數名
作為一個(gè)C++程序員,可能經(jīng)常碰到 __TIME__、__FILE__、__DATE__這樣的宏,它們會(huì )在編譯時(shí),分別轉換為包含編譯時(shí)間、處理的轉換單元名稱(chēng)及當前時(shí)間的字符串?! ?/p> 在最新的ISOC標準中,如大家所知的C99,加入了另一個(gè)有用的、類(lèi)似宏的表達式__func__,其會(huì )報告未修飾過(guò)的(也就是未裁剪過(guò)的)、正在被訪(fǎng)問(wèn)的函數名。請注重,__func__不是一個(gè)宏,因為預處理器對此函數一無(wú)所知;相反,它是作為一個(gè)隱式聲明的常量字符數組實(shí)現的:staticconst char __func__[] = "function-name";
在function-name處,為實(shí)際的函數名。為激活此特性,某些編譯器需要使用特定的編譯標志,請查看相應的編譯器文檔,以獲取具體的資料?! ?/p> 有了它,我們可免去大多數通過(guò)手工修改,來(lái)顯示函數名的苦差事,以上的例子可如下所示進(jìn)行重寫(xiě):voidmyfunc(){cout<<"__FUNCTION__"<<endl;}
官方C99標準為此目的定義的 __func__標識符,確實(shí)值得大家關(guān)注,然而,ISOC++卻不完全支持所有的C99擴展,因此,大多數的編譯器提供商都使用 __FUNCTION__ 取而代之,而 __FUNCTION__通常是一個(gè)定義為 __func__ 的宏,之所以使用這個(gè)名字,是因為它已受到了大多數的廣泛支持?! ?/p> 在VisualStudio2005中,默認情況下,此特性是激活的,但不能與/EP和/P編譯選項同時(shí)使用。請注重在IDE環(huán)境中,不能識別__func__,而要用__FUNCTION__ 代替?! ?/p> Comeau的用戶(hù)也應使用 __FUNCTION__ ,而不是 __func__ ?! ++BuilderX的用戶(hù)則應使用稍稍不同的名字:__FUNC__ ?! CC 3.0及更高的版本同時(shí)支持 __func__和__FUNCTION__ ?! ?/p> 一旦可自動(dòng)獲取當前函數名,你可以定義一個(gè)如下所示顯示任何函數名的函數:void show_name(const char *name){cout<<name<<endl;}voidmyfunc(){show_name(__FUNCTION__); //輸出:myfunc}voidfoo(){show_name(__FUNCTION__); //輸出:foo}
因為_(kāi)_FUNCTION__會(huì )在函數大括號開(kāi)始之后就立即初始化,所以,foo()及myfunc()函數可在參數列表中安全地使用它,而不用擔心重載。
簽名與修飾名
__FUNCTION__ 特性最初是為C語(yǔ)言設計的,然而,C++程序員也會(huì )經(jīng)常需要有關(guān)他們函數的額外信息,在Visual Studio2005中,還支持另外兩種非標準的擴展特性:__FUNCDNAME__ 與 __FUNCSIG__,其分別轉譯為一個(gè)函數的修飾名與簽名。函數的修飾名非常有用,例如,在你想要檢查兩個(gè)編譯器是否共享同樣的ABI時(shí),就可派得上用場(chǎng),另外,它還能幫助你破解那些含義模糊的鏈接錯誤,甚至還可用它從一個(gè)DLL中調用另一個(gè)用C++鏈接的函數。在下例中,show_name()報告了函數的修飾名:voidmyfunc(){show_name( __FUNCDNAME__); //輸出:?myfunc@@YAXXZ}
一個(gè)函數的簽名由函數名、參數列表、返回類(lèi)型、內含的命名空間組成。假如它是一個(gè)成員函數,它的類(lèi)名和const/volatile限定符也將是簽名的一部分。以下的代碼演示了一個(gè)獨立的函數與一個(gè)const成員函數簽名間的不同之處,兩個(gè)函數的名稱(chēng)、返回類(lèi)型、參數完全相同:void myfunc(){show_name(__FUNCSIG__); // void __cdecl myfunc(void)}strUCt S{ void myfunc() const {show_name(__FUNCSIG__); //void__thiscall S::myfunc(void) const} };
本站僅提供存儲服務(wù),所有內容均由用戶(hù)發(fā)布,如發(fā)現有害或侵權內容,請
點(diǎn)擊舉報。