目录
对 Visual Studio 17.4 预览版 3 中的可选复制/移动省略的更改

当 C++ 函数中的关键字后跟非基元类型的表达式时,该 return 语句的执行会将表达式的结果复制到调用函数的返回槽中。为此,将调用非基元类型的复制或移动构造函数。然后,作为退出函数的一部分,调用函数局部变量的析构函数,可能包括关键字后面的表达式中命名的任何变量。returnreturn
C++规范允许编译器直接在调用函数的返回槽中构造返回的对象,省略作为返回的一部分执行的复制或移动构造函数。与大多数其他优化不同,此转换允许对程序的输出产生可观察的影响 - 即,复制或移动构造函数和关联的析构函数被调用一次。

C++ 标准要求在返回值初始化为语句的一部分时(例如,当返回类型的函数返回时)复制或移动省略。Microsoft Visual C++ 编译器始终在需要时对返回语句执行复制和移动省略,而不考虑传递给编译器的标志。此行为保持不变。returnFooreturn Foo()
当返回值是命名变量时,编译器可以省略副本或移动,但不是必需的。该标准仍然要求为命名返回变量定义复制或移动构造函数,即使编译器在所有情况下都省略了构造函数。在 Visual Studio 2022 版本 17.4 预览版 3 之前,当禁用优化(例如使用编译器标志或标记为 的函数)时,编译器将仅执行强制复制和移动省略。使用 theflag,编译器将执行可选的复制或移动 elision,以通过简单的控制流优化函数。
从 Visual Studio 2022 版本 17.4 预览版 3 开始,我们为开发人员提供了与 newcompiler 标志保持一致的选项。默认情况下,当代码使用 theflag、theflag 编译或稍后编译时,将传递 Theflag。传递此标志后,将尽可能执行复制和移动省略。我们希望在将来的版本中默认打开。
/Zc:nrvo/Zc:nrvo/O2/permissive-/std:c++20/Zc:nrvo从 Visual Studio 2022 版本 17.4 预览版 3 开始,还可以使用 theflag 显式禁用可选的复制/移动省略。不可能禁用强制复制/移动省略。
/Zc:nrvo-在 Visual Studio 2022 版本 17.4 预览版 3 中,当启用了可选的复制/移动 elision 时,我们还增加了使用 ,,, oror 更高版本的标志启用复制/移动 elision 的位置数量
Visual Studio 的早期版本 | Visual Studio 17.4 Preview 3 及更高版本 | |
|---|---|---|
| 强制复制/移动省略 | 总是发生。 | 总是发生。 |
| 可选的复制/移动省略,用于返回函数中的命名变量,无需循环或异常处理 | 除非函数具有多个具有重叠生存期的返回符号,或者类型的复制或移动构造函数具有默认参数,否则会发生。/O2 | 不会在下面发生。否则,在 、 或以后发生,或者除非函数具有多个具有重叠生存期的返回符号。/Zc:nrvo-/O2/permissive-/std:c++20/Zc:nrvo |
| 可选的复制/移动省略号,用于在循环中返回命名变量 | 永远不会发生。 | 不会在下面发生。否则,在 、 或以后发生,或者除非函数具有多个具有重叠生存期的返回符号。/Zc:nrvo-/O2/permissive-/std:c++20/Zc:nrvo |
| 可选的复制/移动省略,用于返回具有异常处理的函数中的命名变量 | 永远不会发生。 | 不会在下面发生。否则,在 、 或以后发生,或者除非函数具有多个具有重叠生存期的返回符号。/Zc:nrvo-/O2/permissive-/std:c++20/Zc:nrvo |
| 可选的复制/移动省略,用于在复制或移动构造函数具有其他默认参数时返回命名变量 | 永远不会发生。 | 不会在下面发生。否则,在 、 或以后发生,或者除非函数具有多个具有重叠生存期的返回符号。/Zc:nrvo-/O2/permissive-/std:c++20/Zc:nrvo |
| 用于抛出命名变量的可选复制/移动省略 | 永远不会发生。 | 永远不会发生。 |
可选复制或移动省略的最简单示例是以下函数:
Foo SimpleReturn() {
Foo result;
return result;
}
在这种情况下,如果传递了标志,则早期版本的 MSVC 编译器已省略了复制或移动到返回槽中。在 Visual Studio 2022 版本 17.4 预览版 3 中,如果传递了 orflags 或稍后的标志,则也会省略复制或移动,如果传递了标志,则保留复制或移动。
result/O2/permissive-/std:c++20/Zc:nrvo/Zc:nrvo-从 Visual Studio 2022 版本 17.4 预览版 3 开始,现在在以下其他情况下执行复制/移动省略(如果将 orflags 传递给编译器而 orflags 未传递
对象将在循环的每次迭代开始时正确构造,并在每次迭代结束时销毁。在返回的迭代中,退出函数时不会调用其析构函数。当返回的对象超出该函数的范围时,函数的调用方将销毁返回的对象。resultresult
Foo ReturnInTryCatch() {
try {
Foo result;
return result;
} catch (...) {}
}
如果传递了 orflags 而 orflags 未传递,则对象的复制或移动现在将被省略。现在也妥善处理更复杂的案件,例如:result/O2/permissive-/std:c++20/Zc:nrvo/Zc:nrvo-
int n;
void throwFirstThreeIterations() {
++n;
if (n <= 3) throw n;
}
Foo ComplexTryCatch()
{
Label1:
Foo result;
try {
throwFirstThreeIterations();
return result;
}
catch(...) {
goto Label1;
}
}
该对象将在调用方函数的返回槽中构造,并且在成功返回时不会为其调用任何复制/移动构造函数或析构函数。引发异常时,是否销毁对象取决于将哪些异常处理标志传递给编译器。默认情况下,不会发生堆栈展开,因此不会调用析构函数。但是,如果使用 ,, orflags 启用了堆栈展开异常处理,则会导致调用析构函数,因为它跳转到初始化之前。无论哪种方式,当再次到达表达式时,将在返回槽中再次构造对象
现在正确检测到具有默认参数的复制或移动构造函数仍然是复制或移动构造函数,因此在上述情况下可以省略。具有默认参数的复制构造函数将如下所示:
struct StructWithCopyConstructorDefaultParam {
int X;
StructWithCopyConstructorDefaultParam(int x) : X(x) {}
StructWithCopyConstructorDefaultParam(StructWithCopyConstructorDefaultParam const& original, int defaultParam = 0) :
X(original.X + defaultParam) {
printf("Copy constructor called.\n");
}
};
MSVC 编译器现在在更多情况下执行复制和移动省略,但并非总是可以执行复制/移动省略。要了解为什么会这样,请考虑以下函数:
Foo WhichShouldIReturn(bool condition) {
Foo resultA;
if (condition) {
Foo resultB;
return resultB;
}
return resultA;
}
Copy elision 构造要在返回槽中返回的对象,但在这种情况下,应在返回槽中构造哪个对象?对于要省略的副本,必须在返回槽中构造它。但是,ifis true,将需要在返回槽中构建之前被销毁。无法对这两个路径执行复制省略。resultAreturn resultAconditionresultBresultA
我们目前选择避免在任何路径上都无法复制/移动省略函数中的所有路径上执行可选的复制/移动省略。但是,对内联决策、死代码消除和其他优化的更改可能会更改是否可以复制或移动省略。因此,编写依赖于某些行为的代码来复制/移动命名变量,除非禁用所有可选的复制/移动省略,否则从来都不安全。/Zc:nrvo-
只要启用了堆栈展开异常处理或未引发异常,仍然可以安全地假设每个构造函数调用都有一个匹配的析构函数调用。
完结撒花,下期再见!!!
我需要读入一个包含数字列表的文件。此代码读取文件并将其放入二维数组中。现在我需要获取数组中所有数字的平均值,但我需要将数组的内容更改为int。有什么想法可以将to_i方法放在哪里吗?ClassTerraindefinitializefile_name@input=IO.readlines(file_name)#readinfile@size=@input[0].to_i@land=[@size]x=1whilex 最佳答案 只需将数组映射为整数:@land边注如果你想得到一条线的平均值,你可以这样做:values=@input[x]
我在我的Rails项目中使用Pow和powifygem。现在我尝试升级我的ruby版本(从1.9.3到2.0.0,我使用RVM)当我切换ruby版本、安装所有gem依赖项时,我通过运行railss并访问localhost:3000确保该应用程序正常运行以前,我通过使用pow访问http://my_app.dev来浏览我的应用程序。升级后,由于错误Bundler::RubyVersionMismatch:YourRubyversionis1.9.3,butyourGemfilespecified2.0.0,此url不起作用我尝试过的:重新创建pow应用程序重启pow服务器更新战俘
我正在尝试修改当前依赖于定义为activeresource的gem:s.add_dependency"activeresource","~>3.0"为了让gem与Rails4一起工作,我需要扩展依赖关系以与activeresource的版本3或4一起工作。我不想简单地添加以下内容,因为它可能会在以后引起问题:s.add_dependency"activeresource",">=3.0"有没有办法指定可接受版本的列表?~>3.0还是~>4.0? 最佳答案 根据thedocumentation,如果你想要3到4之间的所有版本,你可以这
我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R
如果我使用ruby版本2.5.1和Rails版本2.3.18会怎样?我有基于rails2.3.18和ruby1.9.2p320构建的rails应用程序,我只想升级ruby的版本,而不是rails,这可能吗?我必须面对哪些挑战? 最佳答案 GitHub维护apublicfork它有针对旧Rails版本的分支,有各种变化,它们一直在运行。有一段时间,他们在较新的Ruby版本上运行较旧的Rails版本,而不是最初支持的版本,因此您可能会发现一些关于需要向后移植的有用提示。不过,他们现在已经有几年没有使用2.3了,所以充其量只能让更
我正在尝试解析一个CSV文件并使用SQL命令自动为其创建一个表。CSV中的第一行给出了列标题。但我需要推断每个列的类型。Ruby中是否有任何函数可以找到每个字段中内容的类型。例如,CSV行:"12012","Test","1233.22","12:21:22","10/10/2009"应该产生像这样的类型['integer','string','float','time','date']谢谢! 最佳答案 require'time'defto_something(str)if(num=Integer(str)rescueFloat(s
我安装了ruby版本管理器,并将RVM安装的ruby实现设置为默认值,这样'哪个ruby'显示'~/.rvm/ruby-1.8.6-p383/bin/ruby'但是当我在emacs中打开inf-ruby缓冲区时,它使用安装在/usr/bin中的ruby。有没有办法让emacs像shell一样尊重ruby的路径?谢谢! 最佳答案 我创建了一个emacs扩展来将rvm集成到emacs中。如果您有兴趣,可以在这里获取:http://github.com/senny/rvm.el
有人知道在发布新版本的Ruby和Rails时收到电子邮件的方法吗?他们有邮件列表,RubyonRails有一个推特,但我不想听到那些随之而来的喧嚣,我只想知道什么时候发布新版本,尤其是那些有安全修复的版本。 最佳答案 从therailsblog获取提要.http://weblog.rubyonrails.org/feed/atom.xml 关于ruby-on-rails-如何在发布新的Ruby或Rails版本时收到通知?,我们在StackOverflow上找到一个类似的问题:
在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList()Obt
我有一个使用SeleniumWebdriver和Nokogiri的Ruby应用程序。我想选择一个类,然后对于那个类对应的每个div,我想根据div的内容执行一个Action。例如,我正在解析以下页面:https://www.google.com/webhp?sourceid=chrome-instant&ion=1&espv=2&ie=UTF-8#q=puppies这是一个搜索结果页面,我正在寻找描述中包含“Adoption”一词的第一个结果。因此机器人应该寻找带有className:"result"的div,对于每个检查它的.descriptiondiv是否包含单词“adoption