1. 程式人生 > >【C++錯誤處理】no matching function for call to transform

【C++錯誤處理】no matching function for call to transform

初學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 > >, )’</CHAR*,></CHAR*,></CHAR*,>

後來查明原因如下——

我們先看看這個函式的定義:

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);