1. 概念
std::unique_ptr
是所谓的智能指针的一种,主要目的是为了解决原生指针安全性不足的弊端。
声明语法:
std::unique_ptr<类型> 变量名称{};
int* a{}; //旧
std::unique_ptr<int> intPtr{}; //新
初始化:
int* b = new int[ 5 ];
std::unique_ptr<int> intPtrB{ new int{ 15 } };
std::cout << *intPtrB << std::endl; // 15
-
C++ 14 及之后的初始化方式:
std::make_unique <类型>()
;std::unique_ptr<int> ptrA{ std::make_unique <int>(5)}; //把指针指向的那块内存初始化为 5; std::cout << *ptrA; std::unique_ptr<int[]> ptrB{ std::make_unique<int[]>(5) }; // 数组初始化为 有 5个元素; ptrB[0] = 250;
2. 智能指针和普通指针的区别:
-
普通指针可以使用
b[0]
访问,智能指针不能使用intPtrB[0]
方式访问。int* b = new int[ 5 ]; std::cout << "b = " << b[0] << std::endl; std::unique_ptr<int> intPtrB{ new int{ 15 } }; std::cout << "intPtrB = " << *intPtrB << std::endl; //std::cout << "intPtrB = " << intPtrB[0] << std::endl; 错
-
std::unique_ptr 指针具有唯一性
;当智能指针A指向一块内存时,其余的智能指针就不能再指向这块内存;std::unique_ptr<int> intPtrA{ new int{ 15 } }; std::unique_ptr<int> intPtrB{ }; // intPtrB = intPtrA; 错 // std::unique_ptr<int> intPtrB{ intPtrA }; 错
3. std::unique_ptr
的用法
3.1. reset()
-
普通指针
delete[] a;
,只释放了内存,没有释放指针; -
智能指针
intPtrA.reset();
即释放了内存,也释放了指针;int* a = new int[5];
delete[] a; //std::unique_ptr ptrA{ std::make_unique (5)};
intPtrA.reset();
3.2. get()
get()
将会返回std::unique_ptr
的指针;
用法:
std::unique_ptr<int> ptrD{ std::make_unique<int>(150) };
int* p = ptrD.get();
std::cout << "ptrD = " << ptrD << std::endl; //ptrD = 0075A4B0
std::cout << "p = " << p << std::endl; //p = 0075A4B0
p
等于ptrD
申请内存时的地址。
注意:
get()
指针赋值只能是赋值给普通指针,智能指针之间还是遵循唯一性
;
int* b = new int{ 5 };
std::unique_ptr<int> intPtrB{ new int{ 15 } };
std::unique_ptr<int> intPtrC{ };
//b = intPtrB; 错
b = intPtrB.get(); 对
// intPtrC = intPtrB.get(); 错
3.3. release()
release
将会返回std::unique_ptr
的指针,并且将std::unique_ptr
设置为nullptr
,但是注意release
并不会释放其占用的内存空间;
用法:
std::unique_ptr<int> ptrD{ std::make_unique<int>(150) };
int* p = ptrD.release();
release()
会把智能指针ptrD
的指针返回,并把ptrD
置为nullptr
,但是其所指向的内存空间还在;p
等于ptrA
申请内存时的地址;
3.4. 指针转移
std::unique_ptr
指针因为具有唯一性,因此不能被复制,但是可以转移。
转移语法:
转移后的指针变量 = std::move(转移前的指针变量);
std::unique_ptr<int> ptrA{ std::make_unique<int>(150) };
std::unique_ptr<int> ptrB{};
ptrB = std::move(ptrA);
- 转移后,
ptrA
被设置为nullptr
。