指向函数签名中的常量智能指针
smart pointer to const in function's signature
我想知道在将 shared_ptr
void f(std::shared_ptr<const Widget> ){}
int main(){
std::shared_ptr<Widget> p;
f(p);
return 0;
}
我假设在这两种情况下,我都在为引用计数的增量和减少付费。
此外,我想知道如果我使用以下签名定义函数f()
,为什么代码无法编译:
void f(shared_ptr<const Widget>& ){}
更让我惊奇的是,这确实是:
void f(const shared_ptr<const Widget>& ){}
为什么按值传递有效?
您的代码由于构造函数重载 (9(smart_ptr
而工作:
template< class Y >
shared_ptr( const shared_ptr<Y>& r ) noexcept;
构造一个共享所管理对象的所有权的shared_ptr 由R.如果 r 不管理任何对象,则也不管理任何对象。这 如果 Y为 不能隐式转换为(直到 C++17(与(自 C++17( T*。
为什么当方法需要shared_ptr<const Widget>&
时它不编译?
如果将签名更改为
void f(shared_ptr<const Widget>& ){}
您不能再在一个步骤中进行转换和传递给该方法,因为临时(由转换产生的临时引用(无法绑定到非常量引用。但是,您仍然可以通过两个步骤完成:
int main(){
std::shared_ptr<Widget> p;
std::shared_ptr<const Widget> p2{p};
// f(p); // error: cannot bind non-const reference to temporary
f(p2); // OK
return 0;
}
是否有一些开销?
关于开销:是的,有一个smart_ptr<const Widget>
正在构建,然后传递给方法(如上面的代码片段中明确显示的那样(。
为什么当方法期望const shared_ptr<const Widget>&
时它再次工作?
关于您的编辑,如果您将签名更改为以下内容,为什么它会再次起作用:
void f(const shared_ptr<const Widget>& ){}
在这种情况下,如果您通过一个shared_ptr<Widget>
,仍然会发生一个转换。但是,现在允许转换产生的临时结果绑定到 const 引用。无论如何,该方法不允许修改它,因此允许通过临时没有危险。
再举一个例子
请注意,不绑定到非常量引用的临时引用是一种罕见的情况,C++可以帮助您避免愚蠢的错误。考虑一下:
void foo(int& x) { x += 2; }
int bar() { return 3; }
int main() { foo(bar()); } // error !
将 r 值传递给需要非常量 l 值引用的函数没有多大意义。您无法观察foo
对bar
返回的值所做的更改。
传递智能指针 @ cpp 核心准则
关于将智能指针传递给功能,请注意,cpp 核心指南对此有一些项目。底线是:如果该方法不参与引用计数(可能是最常见的情况(,则不要传递智能指针,而是传递原始指针。
- 1d 智能指针不适用于语法 (*)++
- #定义c-预处理器常量..我做错了什么
- 用C++中的一个变量定义一个常量
- 什么时候在C++中返回常量引用是个好主意
- 代理对象的常量正确性
- 我想将一个对T类型的非常量左值引用绑定到一个T类型的临时值
- 通过多个头文件使用常量变量
- 在cuda线程之间共享大量常量数据
- 不能在初始值设定项列表中将非常量表达式从类型 'int' 缩小到'unsigned long long'
- 有没有什么方法可以使用一个函数中定义的常量变量,也可以由c++中同一程序中的其他函数使用
- 优先顺序:智能指针和类析构函数
- 是默认情况下分配给char数组常量的值
- 私有类型的静态常量成员
- 类似枚举的计算常量
- 递归模板化函数不能分配给具有常量限定类型"const tt &"的变量"state"
- 指向函数签名中的常量智能指针
- 如何将智能指针向量转换为常量向量,该常量向量包含指向常量的智能指针
- 将常量引用转换为智能指针而不复制
- 智能指针的常量正确性
- 在智能感知中从枚举中计算字符串常量