初學C++哈,不知道這個錯誤是不是很silly,高手輕拍。情況如下: #include #include #include using namespace std; int main (int argc, char * const argv[]){ string str = "Hello"; transform(str.begin(), str.end(), str.begin(), toupper); cout << str << endl; return 0; }程式的意思很簡單,去把Hello都轉換為大寫。 編譯死活不通過: $ g++ -g -Wall strToUpper.cpp -o strToUpper strToUpper.cpp: In function ‘int main(int, char* const*)’: strToUpper.cpp:9: error: no matching function for call to ‘transform(__gnu_cxx::__normal_iterator, std::allocator > >, __gnu_cxx::__normal_iterator, std::allocator > >, __gnu_cxx::__normal_iterator, std::allocator > >, )’後來查明原因如下—— 我們先看看這個函數的定義: template OutIter transform(InIter start, InIter end, OutIter result, Func unaryFunc) 它要求參數和傳回值都要是char。Linux中將toupper實現為一個宏而不是函數: /usr/lib/syslinux/com32/include/ctype.h: /* Note: this is decimal, not hex, to avoid accidental promotion to unsigned */ #define _toupper(__c) ((__c) & ~32) #define _tolower(__c) ((__c) | 32) __ctype_inline int toupper(int __c) { return islower(__c) ? _toupper(__c) : __c; } __ctype_inline int tolower(int __c) { return isupper(__c) ? _tolower(__c) : __c; } 有三種解決方案: 1.因為在全域命名空間中有實現的函數(而不是宏),所以我們明確命名空間,這並不是總奏效,但是在我的g++環境中沒有問題: transform(str.begin(), str.end(), str.begin(), ::toupper); 2.自己寫一個函數出來—wraper inline char charToUpper(char c) { return std::toupper(c); } 3.強制轉化:將toupper轉換為一個傳回值為int,參數只有一個int的函數指標。 transform(str.begin(), str.end(), str.begin(), (int (*)(int))toupper); |