让我们看一个从最差开始到最好的选项列表。我们将在此处列出它们并在下面讨论它们:
transform(cbegin(s), cend(s), begin(s), ::tolower)
transform(cbegin(s), cend(s), begin(s), static_cast<int(*)(int)>(tolower))
transform(cbegin(s), cend(s), begin(s), [](const unsigned char i){ return tolower(i); })
您问题中的代码transform(s.begin(), s.end(), s.begin(), tolower)
将产生如下错误:
没有匹配的调用函数transform(std::basic_string<char>::iterator, std::basic_string<char>::iterator, std::basic_string<char>::iterator, <unresolved overloaded function type>)
您获得“未解析的重载函数类型”的原因是命名空间tolower
中有 2 s std
:
- 该
locale
库定义template <typename T> T tolower(T, const locale&)
- 该
cctype
库定义int tolower(int)
1是davka 提供的解决方案。locale
它通过利用'stolower
未在全局命名空间中定义的事实来解决您的错误。
根据您的情况locale
,tolower
可能值得考虑。您可以在tolower
此处找到 s 的比较:C++ 中的哪个 tolower?
不幸的是1取决于在全局命名空间中定义的cctype
' 。tolower
让我们看看为什么不是这样:
您正确使用#include <cctype>
,因为#include <ctype.h>
在 C++ 中已弃用:http: //en.cppreference.com/w/cpp/header
但是 C++ 标准在 D.3[depr.c.headers]2 中声明了标题中的声明:
未指定这些名称是否首先在命名空间的命名空间范围(3.3.6)内声明或定义,std
然后通过显式使用声明(7.3.3)注入全局命名空间范围
因此,我们可以保证我们的代码独立于实现的唯一方法是使用tolower
from namespace std
。2是David Rodríguez-dribeas 提供的解决方案。它利用了以下事实static_cast
:
用于通过将函数到指针转换为特定类型来消除函数重载的歧义
在我们继续之前,让我评论一下,如果你觉得int (*)(int)
有点困惑,你可以在这里阅读更多关于函数指针语法的内容。
可悲的是, 的输入参数还有另一个问题tolower
,如果它:
不能表示为 unsigned char 且不等于 EOF,行为未定义
您正在使用 a string
,它使用以下类型的元素:char
。char
特别是 7.1.6.2[dcl.type.simple]3的标准状态:
char
类型的对象是表示为有符号量还是无符号量是实现定义的。说明signed
符强制char
对对象进行签名
因此,如果实现将 a 定义char
为表示 a,signed char
那么1和2都会导致与负数对应的所有字符的未定义行为。(如果使用 ASCII 字符编码,则对应于负数的字符是Extended ASCII。)
unsigned char
可以通过在将输入传递给之前将输入转换为 来避免未定义行为tolower
。3使用接受unsigned char
按值的 lambda 来实现这一点,然后将其传递给tolower
隐式转换为int
.
为了保证所有兼容实现上的定义行为,独立于字符编码,您需要使用transform(cbegin(s), cend(s), begin(s), [](const unsigned char i){ return tolower(i); })
或类似的东西。