普通指针到智能指针的转换
int* iPtr = new int(42);
shared_ptr<int> p(iPtr);
智能指针到普通指针的转换
int* pI = p.get();
注意的地方:那就是不要将智能指针与普通指针混用。如果项目允许,坚持使用智能指针,避免原生指针。
智能指针与普通指针需要特别特别特别的小心翼翼,比如以下的情况。
1. 普通指针到智能指针的问题
void f(shared_ptr<int> ptr) //增加引用计数 { // do something... } //销毁ptr,减少引用计数
我们有如下的代码:
int* iPtr = new int(42); f(shared_ptr<int>(iPtr)); int value = *iPtr; // Error! iPtr指针指向的内容已经被释放
因为在这儿,你将普通指针赋予给了一个临时的智能指针,当调用f函数完毕后,此临时智能指针的生命周期结束,然后减少引用计数,归为0,于是,内存释放!
而这儿的更改方法是一直使用智能指针:
auto p = make_shared<int>(42); // 初始化的引用计数为1 f(p); // 拷贝后增加为2,销毁ptr减少1,然后变为1 int value = *p; //引用计数为1
2. 智能指针到普通指针的问题
auto p = make_shared<int>(42); int* iPtr = p.get(); { shared_ptr<int>(iPtr); } int value = *p; // Error! 内存已经被释放
p与iPtr指向了相同的内存,然而通过get方法后,将内存管理权转移给了普通指针。iPtr传递给里面程序块的临时智能指针后,引用计数为1,随后出了作用域,减少为0,释放内存。
额外的注意点:转换
在C++11中,极力推崇完全替代原生指针。而这里面,原生指针使用到了static_cast,dynamic_cast, const_cast的操作,需要用static_pointer_cast, dynamic_pointer_cast, const_pointer_cast对应操作,而并非使用原来的dynamic_cast等。