异常安全和make_unique
Exception safety and make_unique
澄清一下,使用make_unique
只会在表达式中有多个分配时增加异常安全性,而不仅仅是一个,对吗?例如
void f(T*);
f(new T);
是完全异常安全的(就分配和东西而言),而
void f(T*, T*);
f(new T, new T);
不是,对吗?
不仅当你有多个分配时,而且当你可以在不同的地方投掷时。考虑一下:
f(make_unique<T>(), function_that_can_throw());
对:
f(unique_ptr<T>(new T), function_that_can_throw());
在第二种情况下,允许编译器调用(按顺序):
-
new T
-
function_that_can_throw()
-
unique_ptr<T>(...)
显然,如果function_that_can_throw
真的扔了,那么你就会泄漏。 make_unique
防止这种情况。
当然,第二次分配(如您的问题)只是function_that_can_throw()
的特例.
作为一般经验法则,只需使用make_unique
即可使代码保持一致。当您需要unique_ptr
时,它始终是正确的(阅读:异常安全),并且它对性能没有任何影响,因此没有理由不使用它(而实际上不使用它会带来很多陷阱)。
自 C++17 日起,异常安全问题已通过改写 [expr.call] 修复。
参数的初始化,包括每个关联的值计算和副作用,相对于任何其他参数的初始化都是不确定的。
这里的不确定排序意味着一个先于另一个排序,但没有指定哪个。
f(unique_ptr<T>(new T), function_that_can_throw());
只能有两个可能的执行顺序
-
new T
unique_ptr<T>::unique_ptr
function_that_can_throw
-
function_that_can_throw
new T
unique_ptr<T>::unique_ptr
这意味着它现在是异常安全的。
我认为您最好使用std::unique_ptr<T>
来比较实际事物:
void f(std::unique_ptr<T>);
f(std::unique_ptr<T>(new T));
f(std::make_unique<T>());
如果引发异常,这些调用都不会泄漏。然而
void f(std::unique_ptr<T>, std::unique_ptr<T>);
g(std::unique_ptr<T>(new T), std::unique_ptr<T>(new T));
g(std::make_unique<T>(), std::make_unique<T>());
在这种情况下,如果引发异常,显式使用 std::unique_ptr<T>
的版本可能会泄漏(因为编译器可能会在构造任一临时变量之前开始计算 new
-表达式)。
相关文章:
- 为什么 std::unique 不调用 std::sort?
- VS Code "command":"make"与终端窗口中的命令行"make"不同
- 使用 make 编译 MPI,几个命名空间错误,例如"错误:未知类型名称'使用'?
- make 命令如何避免重新编译未更改的源文件?
- MAKE:找不到包含的用户定义的头文件?
- 'make check' GLIBC 运行时的链接问题
- 生成"unique"矩阵
- Qt5 [make -snap] 无法正确编译:进程"/usr/bin/snap"代码 1 退出
- mingw32-make 使用"MinGW Makefiles"生成器跟踪 CMAKE 无法将可执行文件链接到对象库
- 我对 std::unique(算法)C++有问题
- make 命令创建 .file,但不创建应用程序文件
- 如何摆脱导入的 make 项目中的 Eclipse 索引器"Type std::... could not be resolved"错误
- Qt Creator 在执行步骤 "make" 时出现编译错误,-fno-stack-limit
- 如何使用MySQL Connector and Make设置C++项目
- 使用 make 将对象文件放在特定目录中
- 我是 C++ 的新手,我试图调用 make 一个以 2 个类作为其参数的类构造函数
- "Make"失败并出现 Clang 错误 - 如何从 Clang 获得错误?
- 防止 GNU Make 在每次构建时生成 protobuf 代码
- Make zmqpp::socket::connect a std::future
- 链接从命令行转换为Make的库