我偶然发现了一个奇怪的编译问题。我想使用std::for_each
处理字符串列表。以下简化代码说明了问题:
# include <list>
# include <string>
# include <algorithm>
using namespace std ;
void f(wstring & str)
{
// process str here
}
void g(wstring & str, int dummy = 0)
{
// process str here, same as f, just added a second default dummy argument
}
int main(int, char*[])
{
list<wstring> text ;
text.push_back(L"foo") ;
text.push_back(L"bar") ;
for_each(text.begin(), text.end(), f) ; // OK, fine :)
for_each(text.begin(), text.end(), g) ; // Compilation error, complains about
// g taking 2 arguments, but called within std::for_each
// with only one argument.
// ...
return 0 ;
}
我使用MinGW 4.5.2和MSVC10进行了测试,两者都报告了相同的错误消息。最初,我想使用boost::algorithm::trim
作为传递给std::for_each
的处理函数,但我发现它需要两个参数,第一个是必需的(要处理的字符串),第二个是可选的(a locale提供空间字符的定义。)
当使用带有默认参数的函数或方法时,有没有办法在使用std::for_each
(和其他标准算法)时保持清洁?我发现了一种方法可以使它工作,但它不再清晰易懂,因此for
循环开始变得更容易......
# include <list>
# include <string>
# include <algorithm>
# include <boost/bind.hpp>
# include <boost/algorithm/string.hpp>
using namespace std ;
using namespace boost ;
// ... somewhere inside main
list<wstring> text ;
for_each(text.begin(), text.end(), bind(algorithm::trim<wstring>, _1, locale()) ;
// One must deal with default arguments ...
// for_each(text.begin(), text.end(), algorithm::trim<wstring>) would be a real pleasure
感谢您的帮助!
注意:我刚开始学习英语,抱歉有错误:)
答案 0 :(得分:6)
默认参数只是一个代码生成工具,而不是函数签名的一部分,因此您无法真正解决这个问题。你可以将你的函数包装在一个函数对象中,但这正是bind
已经为你做的。
但是,在C ++ 0x中,您可以方便地存储结果(并使用std::bind
)来使代码更具可读性:
auto trimmer = std::bind(boost::algorithm::trim<std::wstring>, std::placeholders::_1, std::locale());
std::for_each(text.begin(), text.end(), trimmer);
(你不想在C ++ 98/03中这样做的原因是bind
的返回类型是相当不雅观的,你不会通过拼写来帮助任何人出。)
或者,再次在C ++ 0x中,您可以使用lambda:
std::for_each(text.begin(), text.end(), [](std::wstring & s){ boost::algorithm::trim<std::wstring>(s); });