- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
以下代码特化了 f() 的两个版本。第一个检测一个 vector 并返回一个迭代器。第二个接受所有其他类型并返回一个拷贝。
这无法在 VC 2010 上编译,GetIter2 中有一个错误,即 GetIter::type 不存在。这仅在使用非 vector 解析对 f() 的调用时才会发生。如果我删除一层类型间接,使用 GetIter 而不是 GetIter2(参见注释行)作为返回类型,那么一切正常。
我想我想知道这是编译器错误还是正确行为。如果这看起来很奇怪,那是因为它减少了我在使用 boost::range_iterator 时遇到的问题,我不能只删除 GetIter2 代表的内容。
#include <vector>
using namespace std;
template<typename T>
struct GetIter {
};
template<typename T>
struct GetIter<vector<T>> {
typedef typename vector<T>::iterator type;
};
template<typename T>
struct GetIter2
{
typedef typename GetIter<T>::type type;
};
template<typename T>
typename enable_if<is_same<T, vector<int>>::value, typename GetIter2<T>::type>::type
//typename enable_if<is_same<T, vector<int>>::value, typename GetIter<T>::type>::type
f(T & t) {
return t.begin();
}
template<typename T>
typename enable_if<!is_same<T, vector<int>>::value, T>::type
f(T & t) {
return t;
}
int main(int argc, char* argv[])
{
vector<int> v(2);
int i = 6;
f(v);
f(i); // error C2039: 'type' : is not a member of 'GetIter<T>'
return 0;
}
编辑:这是我要解决的实际问题。以迭代器作为第二个参数对 copy() 的第二次调用会在 boost::mpl::eval_if_c 对象上导致与上述类似的错误。
#include <vector>
using namespace std;
#include <boost/range.hpp>
#include <boost/tti/has_type.hpp>
BOOST_TTI_TRAIT_HAS_TYPE(has_iterator, iterator)
template<typename InCont, typename Out>
typename enable_if<has_iterator<Out>::value, typename boost::range_iterator<Out>::type>::type
copy(InCont const & in_cont, Out & out_cont)
{
return std::copy(boost::begin(in_cont), boost::end(in_cont), boost::begin(out_cont));
}
template<typename InCont, typename Out>
typename enable_if<!has_iterator<Out>::value, Out>::type
copy(InCont const & in_cont, Out & out_iter)
{
return std::copy(boost::begin(in_cont), boost::end(in_cont), out_iter);
}
int main(int argc, char* argv[])
{
vector<int> v1;
vector<int> v2;
copy(v1, v2);
copy(v1, v2.begin()); // error C2039: 'type' : is not a member of 'boost::mpl::eval_if_c<C,F1,F2>'
return 0;
}
编辑 2:最初的问题在最新版本的 boost::range_iterator 中得到修复。一旦我打了补丁,事情就变得容易了。这是我着陆的内容,使用 boost::has_range_iterator 检查容器:
#include <vector>
using namespace std;
#include <boost/range.hpp>
template<typename InCont, typename Out>
typename boost::range_iterator<Out>::type
copy(InCont const & in_cont, Out & out_cont)
{
return std::copy(boost::begin(in_cont), boost::end(in_cont), boost::begin(out_cont));
}
template<typename InCont, typename Out>
typename enable_if<!boost::has_range_iterator<Out>::value, Out>::type
copy(InCont const & in_cont, Out out_iter)
{
return std::copy(boost::begin(in_cont), boost::end(in_cont), out_iter);
}
int main(int argc, char* argv[])
{
vector<int> v1;
vector<int> v2;
copy(v1, v2);
copy(v1, v2.begin());
return 0;
}
最佳答案
我仍在努力寻找标准的相关部分来解释为什么这不起作用。我认为这与 GetIter2<T>
的替代有关没有失败但随后访问GetIter2<T>::type
然后它确定GetIter<T>::type
不存在。通过制作 GetIter2<T>
模板无法立即替换,您不会收到错误。您可以通过稍微更改其定义来做到这一点:
template<typename T, typename ST = typename GetIter<T>::type>
struct GetIter2
{
typedef ST type;
};
我能够使用以下代码在不修改 boost 类型的情况下让您的真实示例工作:
template<typename InCont, typename Out, typename = typename enable_if<has_iterator<Out>::value>::type>
typename boost::range_iterator<Out>::type
copy(InCont const & in_cont, Out & out_cont)
{
return std::copy(boost::begin(in_cont), boost::end(in_cont), boost::begin(out_cont));
}
关于c++ - SFINAE 多层类型测定,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35677263/
C++11 专家的几个问题。 我正在与 SFINAE 打交道,我遇到了一个奇怪的情况,其中 g++ (4.9.2) 和 clang++ (3.5.0) 的行为不同。 我准备了以下示例代码。很抱歉,我无
这些天我正在试验 SFINAE,有些事情让我很困惑。为什么 my_type_a不能在 my_function 中推导出来的实例化? class my_type_a {}; template clas
我正在尝试复制(我猜)典型的 SFINAE 示例来判断一个类型是否具有特定方法。我的代码基本上是 the one found in the accepted answer of a previous
出于学术原因,我想实现一个示例,如果非类型模板参数满足给定条件,则选择一个模板。例如,我想要一个只为奇数定义的函数。 可以这样做: template struct is_odd: public st
有没有办法检查两个可变参数包的串联是否与第三个可变参数包相同。 template struct ClassOne { } template struct ClassTwo { } template s
为什么以下代码无法编译?为具有特定成员的类型启用模板的最简洁的解决方案是什么?如果模板变量被直接用于初始化它的表达式替换,它也会编译。 #include template constexpr boo
代码: #include using std::nullptr_t; template using nullptr_vt = nullptr_t; struct not_addable{}; tem
我有一个模板类,我想有两个复制 ctor。一个用于平凡类型,另一个用于非平凡类型。 以下代码有效(使用一个拷贝 ctor): template struct MyStruct { MySt
我一直在尝试定义一个辅助类来帮助我使用模板方法,在该方法中我希望为复杂类型和实际类型提供通用实现。 到目前为止,这是我的尝试: #include #include template struct is
这个问题已经有答案了: Officially, what is typename for? [duplicate] (8 个回答) 已关闭 3 年前。 我正在研究现代 C++ 中的 SFINAE,我看
我有一个 std::variants 包含具有不同接口(interface)的对象。目标是如果变体中的对象具有某些方法,则调用它。 我正在尝试制作几个模板装饰器,并寻找一种方法来使用更少的样板和没有宏
我有一个代码,它接受一个函数并根据函数签名执行它,如下所示: template struct Value { int value[Num]; }; struct Executor { t
我一直在尝试定义一个辅助类来帮助我使用模板方法,在该方法中我希望为复杂类型和实际类型提供通用实现。 到目前为止,这是我的尝试: #include #include template struct is
在此视频中https://youtu.be/Vkck4EU2lOU?t=582 “标签调度”和SFINAE作为替代方案出现,允许实现所需模板功能的选择。 正确吗? “标签发送”不是使用SFINAE吗?
我认为下面的代码会编译,因为冲突的重载是 SFINAEd 了。但是编译器(GCC)说:void Foo::bar(Xd) const' cannot be overloaded .有没有简单的方法来修
#define BINDINGTEMPLATE template, int> || std::is_same_v, std::string> || std::is_same_v, char>>> 这样
SFINAE 是否在概念论证中起作用? (也许这里不叫 SFINAE)。例子: template requires std::invocable && // , void>)
在他的演讲现代模板元编程:纲要第 I 部分中,Walter Brown 以他的方式讨论了 enable_if 与 SFINAE 的交互。 在大约 47:40 的谈话中,他被问到一个问题,我无法完全匹配
我正在尝试对同一函数进行两次重载,称为某物。这个函数应该以另一个函数作为参数,并且它应该根据这个另一个函数的返回类型进行重载。到目前为止我有这个: #include #include using
以下代码特化了 f() 的两个版本。第一个检测一个 vector 并返回一个迭代器。第二个接受所有其他类型并返回一个拷贝。 这无法在 VC 2010 上编译,GetIter2 中有一个错误,即 Get
我是一名优秀的程序员,十分优秀!