- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试使用以下代码来理解 move 语义。
#include <iostream>
#include <utility>
using namespace std;
class A
{
public:
int a;
A():a{1}{}
A(A&& rref):a{rref.a}{cout<<"move constructor called"<<endl;}
A(const A& ref):a{ref.a}{cout<<"copy constructor"<<endl;}
};
int main(){
A original; // original object
cout<<"original.a = "<<original.a<< "| address original.a="<< &(original.a)<<endl;
A movedto (std::move(original)); // calls A(A&&)
cout<<"original.a = "<<original.a<< "| address original.a"<< &(original.a)<<endl;
cout<<"movedto.a = "<<movedto.a<<"| address movedto.a"<< &(movedto.a)<<endl;
return 0;
}
给出以下输出
original.a = 1| address original.a=0x7fff1611b6d0
move constructor called
original.a = 1| address original.a0x7fff1611b6d0
movedto.a = 1| address movedto.a0x7fff1611b6e0
可以看出,original.a
和movedto.a
的地址不同,所以对成员a
进行了复制A(A&& rref):a{rref.a}
中的操作。
我知道对于内置类型,move 降级为 copy。我的问题是如果我想劫持(而不是复制)类的一个实例,例如这个实例,我该怎么办。假设我有 100 个(而不是一个)内置类型的成员,使复制变得昂贵。
一个明显的方法是将对象存储在堆上并使用引用语义来传递它。但希望保留值(value)语义并且仍然能够规避复制。
最佳答案
这是不可能的和/或没有意义的。
我们从:
A original; // original object
问题的主要部分是:
What do I do if I want to hijack (not copy) an instance of the class such as this one?
所以这意味着我们最终得到:
A movedto; // new object that has all of original's members
但这里需要注意的是,我们想要“规避复制”并且我们不想使用引用或指针,即“引用语义”,只有“堆栈”或“值语义”。
如果我们希望 movedto
在已经分配的相同内存位置拥有相同的成员,那么我们可以创建对 original
的引用:
A& movedto{original}; // references members at the same memory locations.
但是这个问题的一部分说明我们没有使用引用,因为大概我们希望这个对象有不同的生命周期。因此,如果我们想让 original
的成员“存活”并分配到当前 block 的末尾之后,那么我们会立即发现我们无法控制底层内存。
在这个问题中,original
是一个具有自动存储期限的对象。具有自动存储持续时间的对象会根据其范围自动管理其生命周期。编译器可能使用堆栈来存储它,并且编译器可能使用每次添加对象时向下 move 的堆栈指针,但 C++ 标准没有指定应该如何完成。我们确实知道该标准指定具有自动存储持续时间的对象将按照范围结束时创建的相反顺序销毁。
因此,试图控制创建具有自动存储持续时间的对象的位置没有意义,将此类对象的成员分配给另一个对象也没有意义。内存是自动分配的。
如果我们想重用已经分配为具有自动存储持续时间(堆栈/值语义)的对象的一部分的变量,那么我们使用的内存将在该对象的生命周期结束时被释放。我们必须为此使用动态存储(即“堆”或“引用语义”)。
关于C++ move 语义和内置类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46667460/
我试图在 (C) Python 源代码中找到内置 in 运算符的实现。我在内置函数源代码中搜索过,bltinmodule.c ,但找不到此运算符的实现。我在哪里可以找到这个实现? 我的目标是通过扩展此
我们正在开发一个 shell(学校项目)。我们不理解一种行为。为什么内置函数在重定向时不起作用? 喜欢 cd - | command 不改变目录。 或 export NAME=VALUE | comm
有人问有关如何对列表进行排序的问题。从基本List.Sort()到List.OrderBy()有几种方法。最可笑的是自己动手的SelectionSort。我迅速将其否决,但这使我思考。应用于列表的
我正在尝试使用 C 中内置的 qsort 函数对结构进行排序 typedef struct abc{ long long int fir; long long int sec; }abc; 在
我觉得有一些内置的东西。如果对象为空,我想要默认值(或者特别是 0,我只使用十进制/整数)。是否有编写此函数的内置方法? static int GetDecimalFromObject(object
Java 是否有用于生成和解析文档的内置 XML 库?如果不是,我应该使用哪个第三方? 最佳答案 Sun Java 运行时附带 Xerces 和 Xalan 实现,它们提供解析 XML(通过 DOM
我对 python 的“all”和生成器有以下问题: G = (a for a in [0,1]) all(list(G)) # returns False - as I expected 但是:
我有一些使用 gcc 内部函数的代码。我想包含代码以防缺少内在函数。我该怎么做? #ifdef __builtin_ctzll 不起作用。 最佳答案 使用最新版本的 clang,现在可以使用 __ha
人们常说应该在本地重新声明(某些)Lua 函数,因为这样可以减少开销。但这背后的确切规则/原则是什么?我怎么知道哪些功能应该完成,哪些是多余的?还是应该为每个功能完成,甚至是您自己的功能? 不幸的是,
我想实现以下功能: TestClass values 接受任意数量的 NewClass 对象 只有 NewClass 对象没有完全相同的属性值被添加到TestClass.values 我想出了这个:
我正在尝试编写一个存储过程(使用 SQL Server Management Studio 2008 R2)以从表中检索最大测量值。这似乎是一件容易的事,所以我写了一个简短的存储过程来获取 MAX。但
我刚写了我的第一个Electron应用程序。现在,我正在尝试通过electron-packager构建它。我的package.json看起来像这样: { "name": "pixelcast",
我正在寻找在 WPF 应用程序中使用的“安全”字体系列列表 - 应该安装在所有能够运行 WPF 的客户端机器上的字体系列。 Silverlight 有一个明确定义的列表( listed on MSDN
好吧,(在写了几次之后)发现System.Windows.Controls命名空间中已经有一个BooleanToVisibilityConverter,这真是一个惊喜。 可能还有更多这样隐藏的节省时间
在我的 gradle 构建文件中,我有以下插件 block plugins { `java-library` jacoco checkstyle } 这些都没有指定版本,但一切
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 3 年前。 Improve this ques
10 implementations String#reverse 已根据每个浏览器进行分析。 自 2011 年以来已对这些实现进行了解释。 当 ES6 出现时,有很多代码变得更加优雅和性能。 关于
在 Julia 包 BenchmarkTools 中,有一些像 @btime、@belapse 这样的宏对我来说似乎是多余的,因为 Julia 内置了@time、@elapse 宏。在我看来,这些宏服
我正在尝试编写一个简单的 LLVM 通行证,其目标如下: 查找所有 call指示。 在被调用函数中插入我编写的外部函数。 例如,考虑我有以下示例程序: #include #include int
我理解 'a) -> (rhs:'a -> 'a) -> 'a 在我感兴趣的情况下,我经常发现自己想要类似 (lhs:'a -> 'b) -> (rhs:'c -> 'b) -> 'b 的东西在侧面
我是一名优秀的程序员,十分优秀!