gpt4 book ai didi

c++ - 如何在 lambda 中使用模板类型参数?

转载 作者:塔克拉玛干 更新时间:2023-11-02 23:06:32 26 4
gpt4 key购买 nike

是否可以在本地匿名函数中使用来自周围模板函数的模板类型参数?我很确定我不能声明模板 lambda ...

例如,我将如何着手做这样的事情:

template <typename T>
void TrimString(std::basic_string<T>& str, const std::locale& loc = std::locale())
{
// std::isspace as lambda unary predicate?
auto fn = [&loc](T c){ return std::use_facet<std::ctype<T>>(loc).is(std::ctype_base::space, c); };
// trim right
str.erase(std::find_if(str.rbegin(), str.rend(), std::not1(fn)).base(), str.end());
// trim left
str.erase(str.begin(), std::find_if(str.begin(), str.end(), std::not1(fn)));
}

目前这会产生以下错误:

error C2039: 'argument_type' : is not a member of '`anonymous-namespace'::<lambda0>'

这是有道理的,因为 lambda 不知道来自周围模板函数的参数 T

我使用 VS2010 和 gcc 4.7,但我不想使用 boost。

有什么想法吗?

编辑:看来我错误地认为问题出在模板参数本身。相反,它是使用 lambda 函数编译的 std::not1。这是更详细的错误输出:

error C2039: 'argument_type' : is not a member of '`anonymous-namespace'::<lambda0>'
: see declaration of '`anonymous-namespace'::<lambda0>'
: see reference to class template instantiation 'std::unary_negate<_Fn1>' being compiled
with
[
_Fn1=`anonymous-namespace'::<lambda0>
]
: see reference to function template instantiation 'void TrimString<char>(std::basic_string<_Elem,_Traits,_Ax> &,const std::locale &)' being compiled
with
[
_Elem=char,
_Traits=std::char_traits<char>,
_Ax=std::allocator<char>
]

如果参数是函数类型,是否需要显式声明参数类型?我不确定我做错了什么......

答案:

选项 1:如果我不使用 std::not1 而是取反 lambda 中的返回值,我会毫无问题地得到相同的行为。

auto fn = [&loc](T c){ return !std::use_facet<std::ctype<T>>(loc).is(std::ctype_base::space, c); };

选项 2:由于 lambda 不再等同于 std::isspace 作为一元谓词的行为方式,因此函数对象构造函数转换也可以解决问题。

str.erase(std::find_if(str.rbegin(), str.rend(), std::not1(std::function<bool(T)>(fn))).base(), str.end());

最佳答案

问题不是由在 lambda 内部使用模板参数引起的,因为在构造 lambda 时参数已经解析为类型。

问题是你定义的lambda不能和std::not1组合,这需要一个 std::unary_function<argument_type,return_type> 作为参数.

解决问题最简单的方法就是不用std::not1 ,取而代之的是否定 lambda 表达式中的断言权:

auto fn = [&loc](T c){ return !std::use_facet<std::ctype<T>>(loc).is(std::ctype_base::space,c); };

然后编译并使用 GCC 4.7.0 的完整代码变为:

#include <string>
#include <algorithm>
#include <locale>
#include <iostream>

template <typename T>
void TrimString(std::basic_string<T>& str, const std::locale& loc = std::locale())
{
auto fn = [&loc](T c){ return !std::use_facet<std::ctype<T>>(loc).is(std::ctype_base::space,c); };

str.erase(std::find_if(str.rbegin(), str.rend(),fn).base(), str.end());
str.erase(str.begin(), std::find_if(str.begin(), str.end(), fn));
}

int main() {
std::basic_string<char> s(" hello ");
TrimString(s);
std::cout << s << std::endl;
return 0;
}

这输出

hello

正如预期的那样。

关于c++ - 如何在 lambda 中使用模板类型参数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11130168/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com