- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
最近,我有以下情况
struct data {
std::vector<int> V;
};
data get_vector(int n)
{
std::vector<int> V(n,0);
return {V};
}
这段代码的问题在于,当创建结构体时会发生复制,解决方案是编写 return {std::move(V)}
是否有 linter 或代码分析器可以检测此类虚假复制操作? cppcheck、cpplint 和 clang-tidy 都无法做到这一点。
编辑:让我的问题更清楚的几点:
最佳答案
我相信你的观察是正确的,但解释是错误的!
返回值不会发生复制,因为每个正常的聪明编译器都会使用 (N)RVO在这种情况下。从 C++17 开始,这是强制性的,因此您无法通过从函数返回本地生成的 vector 来看到任何拷贝。
好的,让我们来玩一下 std::vector
以及在构造过程中或逐步填充过程中会发生什么。
首先,让我们生成一种数据类型,使每个拷贝或移动都可见,如下所示:
template <typename DATA >
struct VisibleCopy
{
private:
DATA data;
public:
VisibleCopy( const DATA& data_ ): data{ data_ }
{
std::cout << "Construct " << data << std::endl;
}
VisibleCopy( const VisibleCopy& other ): data{ other.data }
{
std::cout << "Copy " << data << std::endl;
}
VisibleCopy( VisibleCopy&& other ) noexcept : data{ std::move(other.data) }
{
std::cout << "Move " << data << std::endl;
}
VisibleCopy& operator=( const VisibleCopy& other )
{
data = other.data;
std::cout << "copy assign " << data << std::endl;
}
VisibleCopy& operator=( VisibleCopy&& other ) noexcept
{
data = std::move( other.data );
std::cout << "move assign " << data << std::endl;
}
DATA Get() const { return data; }
};
现在让我们开始一些实验:
using T = std::vector< VisibleCopy<int> >;
T Get1()
{
std::cout << "Start init" << std::endl;
std::vector< VisibleCopy<int> > vec{ 1,2,3,4 };
std::cout << "End init" << std::endl;
return vec;
}
T Get2()
{
std::cout << "Start init" << std::endl;
std::vector< VisibleCopy<int> > vec(4,0);
std::cout << "End init" << std::endl;
return vec;
}
T Get3()
{
std::cout << "Start init" << std::endl;
std::vector< VisibleCopy<int> > vec;
vec.emplace_back(1);
vec.emplace_back(2);
vec.emplace_back(3);
vec.emplace_back(4);
std::cout << "End init" << std::endl;
return vec;
}
T Get4()
{
std::cout << "Start init" << std::endl;
std::vector< VisibleCopy<int> > vec;
vec.reserve(4);
vec.emplace_back(1);
vec.emplace_back(2);
vec.emplace_back(3);
vec.emplace_back(4);
std::cout << "End init" << std::endl;
return vec;
}
int main()
{
auto vec1 = Get1();
auto vec2 = Get2();
auto vec3 = Get3();
auto vec4 = Get4();
// All data as expected? Lets check:
for ( auto& el: vec1 ) { std::cout << el.Get() << std::endl; }
for ( auto& el: vec2 ) { std::cout << el.Get() << std::endl; }
for ( auto& el: vec3 ) { std::cout << el.Get() << std::endl; }
for ( auto& el: vec4 ) { std::cout << el.Get() << std::endl; }
}
我们可以观察到什么:
示例 1)我们从初始化列表创建一个 vector ,也许我们期望看到 4 次构造和 4 次移动。但我们得到了 4 份!这听起来有点神秘,但原因是初始化列表的实现!简而言之,不允许从列表中移动元素,因为列表中的迭代器是 const T* ,这使得无法从列表中移动元素。有关此主题的详细答案可以在这里找到:initializer_list and move semantics
示例 2)在本例中,我们得到一个初始构造和该值的 4 个拷贝。这没什么特别的,也是我们可以期待的。
示例 3)同样在这里,我们按照预期进行了 build 和一些行动。通过我的 STL 实现, vector 每次都会增长 2 倍。因此,我们看到第一个构造,另一个构造,并且由于 vector 大小从 1 调整为 2,所以我们看到第一个元素的移动。在添加 3 时,我们看到大小从 2 调整为 4,这需要移动前两个元素。一切都如预期!
示例 4)现在我们先预留空间,稍后再填补。现在我们没有拷贝,也没有移动了!
在所有情况下,通过将 vector 返回给调用者,我们根本看不到任何移动或复制! (N)RVO 正在发生,此步骤无需采取进一步操作!
回到你的问题:
"How to find C++ spurious copy operations"
如上所示,您可以在中间引入一个代理类以用于调试目的。
在许多情况下,将复制者设为私有(private)可能不起作用,因为您可能有一些想要的拷贝和一些隐藏的拷贝。如上所述,只有示例 4 的代码可以与私有(private)复制者一起使用!我无法回答这个问题,如果示例 4 是最快的,因为我们会以和平来实现和平。
抱歉,我无法在此处提供查找“不需要的”拷贝的通用解决方案。即使您挖掘代码以调用 memcpy
,您也不会找到所有内容,因为 memcpy
也会被优化掉,您会直接看到一些汇编指令在不调用的情况下完成这项工作到您的库 memcpy
函数。
我的提示是不要把注意力集中在这么小的问题上。如果您确实遇到性能问题,请使用分析器进行测量。有太多潜在的性能 killer ,因此在虚假的 memcpy
使用上投入大量时间似乎不是一个值得的想法。
关于c++ - 如何发现C++虚假复制操作?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59560436/
我正在努力做到这一点 在我的操作中从数据库获取对象列表(确定) 在 JSP 上打印(确定) 此列表作为 JSP 中的可编辑表出现。我想修改然后将其提交回同一操作以将其保存在我的数据库中(失败。当我使用
我有以下形式的 Linq to Entities 查询: var x = from a in SomeData where ... some conditions ... select
我有以下查询。 var query = Repository.Query() .Where(p => !p.IsDeleted && p.Article.ArticleSections.Cou
我正在编写一个应用程序包,其中包含一个主类,其中主方法与GUI类分开,GUI类包含一个带有jtabbedpane的jframe,它有两个选项卡,第一个选项卡包含一个jtable,称为jtable1,第
以下代码产生错误 The nested query is not supported. Operation1='Case' Operation2='Collect' 问题是我做错了什么?我该如何解决?
我已经为 HA redis 集群(2 个副本、1 个主节点、3 个哨兵)设置了本地 docker 环境。只有哨兵暴露端口(10021、10022、10023)。 我使用的是 stackexchange
我正在 Desk.com 中构建一个“集成 URL”,它使用 Shopify Liquid 模板过滤器语法。对于开始日期为 7 天前而结束日期为现在的查询,此 URL 需要包含“开始日期”和“结束日期
你一定想过。然而情况却不理想,python中只能使用类似于 i++/i--等操作。 python中的自增操作 下面代码几乎是所有程序员在python中进行自增(减)操作的常用
我需要在每个使用 github 操作的手动构建中显示分支。例如:https://gyazo.com/2131bf83b0df1e2157480e5be842d4fb 我应该显示分支而不是一个。 最佳答
我有一个关于 Perl qr 运算符的问题: #!/usr/bin/perl -w &mysplit("a:b:c", /:/); sub mysplit { my($str, $patt
我已经使用 ArgoUML 创建了一个 ERD(实体关系图),我希望在一个类中创建两个操作,它们都具有 void 返回类型。但是,我只能创建一个返回 void 类型的操作。 例如: 我能够将 book
Github 操作仍处于测试阶段并且很新,但我希望有人可以提供帮助。我认为可以在主分支和拉取请求上运行 github 操作,如下所示: on: pull_request push: b
我正在尝试创建一个 Twilio 工作流来调用电话并记录用户所说的内容。为此,我正在使用 Record,但我不确定要在 action 参数中放置什么。 尽管我知道 Twilio 会发送有关调用该 UR
我不确定这是否可行,但值得一试。我正在使用模板缓冲区来减少使用此算法的延迟渲染器中光体积的过度绘制(当相机位于体积之外时): 使用廉价的着色器,将深度测试设置为 LEQUAL 绘制背面,将它们标记在模
有没有聪明的方法来复制 和 重命名 文件通过 GitHub 操作? 我想将一些自述文件复制到 /docs文件夹(:= 同一个 repo,不是远程的!),它们将根据它们的 frontmatter 重命名
我有一个 .csv 文件,其中第一列包含用户名。它们采用 FirstName LastName 的形式。我想获取 FirstName 并将 LastName 的第一个字符添加到它上面,然后删除空格。然
Sitecore 根据 Sitecore 树中定义的项目名称生成 URL, http://samplewebsite/Pages/Sample Page 但我们的客户有兴趣降低所有 URL(页面/示例
我正在尝试进行一些计算,但是一旦我输入金额,它就会完成。我只是希望通过单击按钮而不是自动发生这种情况。 到目前为止我做了什么: Angular JS - programming-fr
我的公司创建了一种在环境之间移动文件的复杂方法,现在我们希望将某些构建的 JS 文件(已转换和缩小)从一个 github 存储库移动到另一个。使用 github 操作可以实现这一点吗? 最佳答案 最简
在我的代码中,我创建了一个 JSONArray 对象。并向 JSONArray 对象添加了两个 JSONObject。我使用的是 json-simple-1.1.jar。我的代码是 package j
我是一名优秀的程序员,十分优秀!