- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
所以给定以下两个函数:
int rvalue();
int& lvalue();
以下是有效的:
std::tuple<int&, int> x = appropriate_fn(lvalue(), rvalue());
我是这样想的:
template <typename T, typename...Ts>
auto make_comparible(T const& arg, Ts&&...args)
{
return std::make_tuple(T(arg), make_comparible(args...));
}
template <typename T, typename...Ts>
auto make_comparible(T& arg, Ts&&...args)
{
return std::make_tuple<T&>(arg, make_comparible(args...));
}
template <typename T>
auto make_comparible(T const& arg)
{
return std::tuple<T>(T(arg));
}
template <typename T>
auto make_comparible(T& arg)
{
return std::tuple<T&>(arg);
}
但是我可以看到三个问题。
这不是一个简单的std::tuple
,而是一个嵌套的。仔细想想,这可能不是问题,因为我只想对其进行比较(小于、等于)并且应该仍然有效。
这不区分临时引用和常量引用。这有点烦人,但我看不出有任何解决办法。
最重要的是,它不起作用。鉴于以下情况:
std::tuple<int, std::tuple<int&>> x = make_comparible(rvalue(), lvalue());
std::tuple<int&, std::tuple<int>> y = make_comparible(lvalue(), rvalue());
第一个有效,但第二个给出错误,因为 make_comparible()
正在返回 std::tuple<int, std::tuple<int&>>
而不是 std::tuple<int&, std::tuple<int>>
. <知识库> Demo
那么,我所要求的是可能的,还是一个白日梦?
用例是我想在一个类中定义一个函数,该函数将返回一个元组(或其他类似类型),这不会导致悬空指针/引用,并且易于使用。
好的,看完之后C++ and Beyond 2012: Scott Meyers - Universal References in C++11 ,看起来在通用引用上重载几乎总是一个错误。但是当我试图区分左值和右值时,无论常量如何,这将是正确的方法。
如果我声明用于左值的重载,我可以使用重载来获得仅绑定(bind)到右值的通用引用。我也忘记使用 std::forward()
这对我来说是一个脑放屁。我早该知道的。
#include <iostream>
#include <tuple>
// forward declarations
template <typename T, typename...Ts>
decltype(auto) make_comparible(T const& arg, Ts&&...args);
template <typename T, typename...Ts>
decltype(auto) make_comparible(T& arg, Ts&&...args);
template <typename T>
decltype(auto) make_comparible(T&& arg);
template <typename T>
decltype(auto) make_comparible(T const& arg);
template <typename T>
decltype(auto) make_comparible(T& arg);
// rvalue
template <typename T, typename...Ts>
decltype(auto) make_comparible(T&& arg, Ts&&...args)
{
std::cout << "rvalue ";
// want to copy, so do not use std::move()
return std::make_tuple(arg, make_comparible(std::forward<Ts>(args)...));
}
// lvalue const
template <typename T, typename...Ts>
decltype(auto) make_comparible(T const& arg, Ts&&...args)
{
std::cout << "lvalue const ref ";
// This is a reference, so store as a reference
return std::make_tuple<T const&>(arg, make_comparible(std::forward<Ts>(args)...));
}
// lvalue
template <typename T, typename...Ts>
decltype(auto) make_comparible(T& arg, Ts&&...args)
{
std::cout << "lvalue ref ";
// This is a reference, so store as a reference
return std::make_tuple<T&>(arg, make_comparible(std::forward<Ts>(args)...));
}
// rvalue
template <typename T>
decltype(auto) make_comparible(T&& arg)
{
std::cout << "rvalue ";
// want to copy, so do not use std::move()
return std::tuple<T>(arg);
}
// lvalue const
template <typename T>
decltype(auto) make_comparible(T const& arg)
{
std::cout << "lvalue const ref ";
// This is a reference, so store as a reference
return std::tuple<T const&>(arg);
}
// lvalue
template <typename T>
decltype(auto) make_comparible(T& arg)
{
std::cout << "lvalue ref ";
// This is a reference, so store as a reference
return std::tuple<T&>(arg);
}
int var = 5;
int rvalue() { return 4; }
int& lvalue() { return var; }
int const& const_lvalue() { return var; }
int main()
{
// expect output "rvalue lvalue ref", OK
std::tuple<int, std::tuple<int&>> x = make_comparible(rvalue(), lvalue());
std::cout << std::endl;
// expect output "rvalue lvalue const ref", OK
std::tuple<int, std::tuple<int const&>> y = make_comparible(rvalue(), const_lvalue());
std::cout << std::endl;
// expect output "lvalue ref lvalue const ref rvalue", OK
make_comparible(lvalue(), const_lvalue(), rvalue());
// But this doesn't work. Type returned was std::tuple<int, std::tuple<int, std::tuple<int> > >. WHY?
std::tuple<int&, std::tuple<int const&, std::tuple<int>>> z = make_comparible(lvalue(), const_lvalue(), rvalue());
std::cout << std::endl;
return 0;
}
所以代码路径是正确的。但是返回的类型是错误的。我得到一个 std::tuple<int, std::tuple<int, std::tuple<int>>>
而不是 std::tuple<int&, std::tuple<int const&, std::tuple<int>>>
.为什么?
最佳答案
如果我理解正确你想要什么,你可以使用 std::reference
(包装一个左值引用,以便 std::make_tuple()
产生std::tuple
,并在相应位置引用)和 std::forward
,从可变参数列表中获取正确类型的引用。
所以你可以写几个转换函数
int rVal ()
{ return 0; }
int & lVal ()
{ static int val { 1 }; return val; }
和make_comparable()
成为
template <typename ... Ts>
auto make_comparible (Ts && ... args)
{ return std::make_tuple(convert(std::forward<Ts>(args))...); }
如果你可以使用 C++14/C++17(auto
返回类型),或者
template <typename ... Ts>
auto make_comparible (Ts && ... args)
-> decltype(std::make_tuple(convert(std::forward<Ts>(args))...))
{ return std::make_tuple(convert(std::forward<Ts>(args))...); }
或者也(更简单)
template <typename ... Ts>
auto make_comparible (Ts && ... args)
-> decltype(std::make_tuple(convert(std::forward<Ts>(args))...))
{ return { convert(std::forward<Ts>(args))... }; }
如果您必须使用 C++11(auto
加上 decltype()
;丑陋但有效)。
以下是一个完整的工作 (C++14) 示例。
#include <tuple>
#include <functional>
int rVal ()
{ return 0; }
int & lVal ()
{ static int val { 1 }; return val; }
template <typename T>
std::reference_wrapper<T> convert (T & t)
{ return t; }
template <typename T>
T convert (T && t)
{ return std::move(t); }
template <typename ... Ts>
auto make_comparible (Ts && ... args)
{ return std::make_tuple(convert(std::forward<Ts>(args))...); }
int main ()
{
auto t = make_comparible(rVal(), lVal());
static_assert(std::is_same<std::tuple<int, int&>, decltype(t)>{}, "!");
}
关于c++ - 有没有办法将左值和右值列表分别转换为具有引用类型和完整类型的元组?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46894411/
我想知道最终用户按下了什么,所以我使用了 getch() 。 如果用户按右,我可以获得0xE0 0x4D。 如果用户按下Ctrl+右,我可以获得0xE0 0x47。 如果用户按下Shift+右,我可以
我已经构建了一个应用程序来搜索我的位置。 这是代码 var map; var gdir; var geocoder = null; var addressMarker; function init
我想为我的元素设计布局 View 。布局 View 在左 Angular 和右 Angular (同一行)有一个图像,将有 2 行单词,一行在第 1 行,另一行在第 2 行。我该如何实现? It
我有一个很长的线性(分支不多)流程图,在 graphviz 中显示为要么太高而无法放在单个页面上,要么太宽(如果方向是从左到右) 是否有一种简单的方法可以让 graphviz 以从左到右,然后向下,然
我一直摸不着头脑,但运气不好。设计器有一个包含 3 栏的站点、两个侧边栏和一个主要内容区域。 专为桌面设计,左栏、主要内容、右栏。但是,在较小的设备上,我们希望首先堆叠主要内容。 所以通常情况下,你可
我想要从上到下和从左到右组织的 css block 。 为了更好地解释这是一张图片,其中包含我到目前为止所获得的内容以及我希望使用 CSS 实现的内容: 代码如下: HTML: 1 2 3 4 5
当我问this question时,答案之一(现已删除)建议Either类型对应Curry-Howard correspondence中的XOR而不是OR,因为它不能同时是Left和Right。 真相
如果一行中六个观察值中至少有三个是 != NA,我想计算该行的平均值。如果存在四个或更多 NA,则平均值应显示为 NA。 给出平均值的例子,忽略了 NA: require(dplyr) a % mut
我有一个由 9 列组成的数据框,其中包含一个因素 list 。每行可以填充所有 9 列(因为在该行中包含 9 个“事物”),但大多数没有(大多数有 3-4 个)。列也不是特定的,就像第 1 列和第 3
这是我第一次尝试使用 R 构建函数。基本上我的预期目标如下。 使用 RoogleVision 包与 Google Cloud Vision API 通信 函数遍历目录中的图片 从每张图片的 Googl
使用: mean (x, trim=0.05) 从分布的每一侧移除 2.5%,这对于对称的双尾数据来说很好。但是如果我有一个尾部或高度不对称的数据,我希望能够只删除分布的一侧。有没有这个功能,还是我自
我想保留重复的列,并删除唯一的列。这些列将具有相同的值,但名称不同。 x1 = rnorm(1:10) x2 = rnorm(1:10) x3 = x1 x4 = rnorm(1:10) x5 = x
是否可以使WPF工具栏中的元素的Right水平对齐方式正确? 我尝试将内部元素添加到Grid中,并将ColumnDefinition分配给Left / Right。我
datatable(head(iris)) 如何将我的列居中,使其位于我的列名称的正下方? 最佳答案 您可以使用options 下的columnDefs 自变量。将 className 设置为 dt-
我是 R 的新手,但我正在尝试在 R 中制作滑动窗口。 使用循环我可以像这样,但这变得非常低效。 results=c(1:7) letters=c("A","B","C","D","E","F","G
假设我有这个 .txt 文件: here is line 1 here is line 2 here is line 3 here is line 4 我想将此字符串粘贴到第 3 行和第 4 行之间:
假设我有这个 .txt 文件: here is line 1 here is line 2 here is line 3 here is line 4 我想将此字符串粘贴到第 3 行和第 4 行之间:
我想知道我的环境中有什么类型的对象。 我可以像这样显示谁在那里: ls() 但是运行类似的东西 sapply(ls(), class) (显然)不会告诉我们我们拥有什么类型(类)的对象(函数、数字、因
我想创建一个带有水平标签的树状图,但让叶子根据它们的高度悬挂,而不是仅仅下降到图的边缘。 例子: par(mfrow = c(1,2)) hc <- hclust(dist(USArrests), "
我的 CSS 中有一个元素,如下所示 .xyz{ position:absolute; left:50%; } 现在正如预期的那样,当我减小浏览器窗口的宽度时,这个元素向左移动
我是一名优秀的程序员,十分优秀!