C++11 智能指针
C++ 11标准库引入了几种智能指针
unique_ptr
shared_ptr
weak_ptr
C++内存管理机制是当一个变量或对象从作用域过期的时候就会从内存中将他干掉。但是如果变量只是一个指针的话,干掉的就只是指针自身,而指针所指向的内存还是存在的,这就是所谓的内存泄漏。智能指针的出现就是为了解决这个问题。
智能指针的大体原理:当智能指针对象过期的时候,会执行智能指针对象的析构函数,在这里我们选择进行对所指堆对象的清除操作。
智能指针的头文件是
#include<memory>
定义在标准库命名空间 std 当中
其中:
unique_ptr是不允许复制操作的,所以一个unique_ptr不能直接赋值给另一个unique_ptr或者其他类型的智能指针。unique_ptr重载了->运算符可以直接寻找对应对象的
内容。它自始至终都是一个对象对应一个智能指针,绝对不允许把同一个对象赋值给两个unique_ptr指针,同时,一旦一个对象赋值给一个unique_ptr那么将不会有别的智能指针去被这个对象赋值。
如果我们想更改对象的赋值,我们智能使用std::move方法,这个方法废掉了原来的unique_ptr,使之变为无用的智能指针,然后赋值给一个新的智能指针。
shared_ptr则相反,它可以有很多个实例管理同一个堆对象,每增加一个实例,引用计数+1,过期一个实例,引用计数-1。当引用计数为0的时候释放堆内存。
weak_ptr不能直接用->获取对象的内容,需要调用lock()方法获得对象。此外使用方法和shared_ptr类似,但是不参与引用计数的计算。
接下来打算用一组实例来直观的反映一下智能指针的使用
#include<iostream>
#include<memory>
#include<string>
#include<functional>
using namespace std;
class poco_one
{
public:
poco_one(string n);
~poco_one();
public:
void showDesc();
private:
string name;
};
void testUniquePtr();
void testSharePtr();
void testWeakPtr();
int main()
{
testUniquePtr();
system("pause");
testSharePtr();
cout << "spc1 已经过期" <<endl;
system("pause");
testWeakPtr();
system("pause");
}
void testSharePtr()
{
cout << "测试share智能指针" << endl;
shared_ptr<poco_one> spc1(new poco_one("lily"));
cout << "当前引用计数为 : " << spc1.use_count() <<endl;
function<void(void)> func = [&]()
{
shared_ptr<poco_one> spc2 = spc1;
cout << "当前引用计数为 : " << spc1.use_count() << endl;
cout << "spc2 即将过期" <<endl;
};
func();
cout << "当前引用计数为 : " << spc1.use_count() <<endl;
cout << "spc2已过期"<<endl;
}
void testWeakPtr()
{
cout << "测试weak智能指针" << endl;
shared_ptr<poco_one> spc1(new poco_one("blaze"));
cout << "当前引用计数为 : " << spc1.use_count() << endl;
shared_ptr<poco_one> spc2 = spc1;
cout << "当前引用计数为 : " << spc1.use_count() << endl;
weak_ptr<poco_one> wpc = spc2;
cout << "当前引用计数为 : " << spc1.use_count() << endl;
wpc.lock()->showDesc();
}
void testUniquePtr()
{
cout << "测试unique智能指针"<<endl;
unique_ptr<poco_one> pcone1(new poco_one("rock"));
//unique_ptr<poco_one> pcone2 = pcone1; //无效 不可复制
pcone1->showDesc();
unique_ptr<poco_one> pcone3 = std::move(pcone1);
pcone1->showDesc();
pcone3->showDesc();
pcone3.reset();
pcone3->showDesc();
pcone1.reset(); //已经无效
unique_ptr<poco_one> pcone4;
pcone4->showDesc();
}
poco_one::poco_one(string n)
{
cout << "poco_one 对象已经被初始化" << endl;
name = n;
}
poco_one::~poco_one()
{
cout << "poco_one 对象即将被销毁" << endl;;
}
void poco_one::showDesc()
{
if (&name == NULL)
{
cout << "本对象已经被干掉或者不存在" << endl;
return;
}
cout << name + ":show something"<< endl;
}
其执行结果为:
测试unique智能指针 poco_one 对象已经被初始化 rock:show something 本对象已经被干掉或者不存在 rock:show something poco_one 对象即将被销毁 本对象已经被干掉或者不存在 本对象已经被干掉或者不存在 请按任意键继续. . . 测试share智能指针 poco_one 对象已经被初始化 当前引用计数为 : 当前引用计数为 : spc2 即将过期 当前引用计数为 : spc2已过期 poco_one 对象即将被销毁 spc1 已经过期 请按任意键继续. . . 测试weak智能指针 poco_one 对象已经被初始化 当前引用计数为 : 当前引用计数为 : 当前引用计数为 : blaze:show something poco_one 对象即将被销毁 请按任意键继续. . .
注:
C++类的成员方法即便不实例化对象也可以调用,同时智能指针也可以调用。但是如果动态成员变量没有初始值的话就会很危险!
lambda表达式如果用[=]复制的方式传参,那么这个复制出来的值要在lambda表达式所在的方法结束后才会过期,而不是lambda方法本身结束后过期。
C++11 智能指针的更多相关文章
- c++11 智能指针 unique_ptr、shared_ptr与weak_ptr
c++11 智能指针 unique_ptr.shared_ptr与weak_ptr C++11中有unique_ptr.shared_ptr与weak_ptr等智能指针(smart pointer), ...
- C++11——智能指针
1. 介绍 一般一个程序在内存中可以大体划分为三部分——静态内存(局部的static对象.类static数据成员以及所有定义在函数或者类之外的变量).栈内存(保存和定义在函数或者类内部的变量)和动态内 ...
- C++11智能指针之std::unique_ptr
C++11智能指针之std::unique_ptr uniqut_ptr是一种对资源具有排他性拥有权的智能指针,即一个对象资源只能同时被一个unique_ptr指向. 一.初始化方式 通过new云 ...
- 【C++11新特性】 C++11智能指针之weak_ptr
如题,我们今天要讲的是C++11引入的三种智能指针中的最后一个:weak_ptr.在学习weak_ptr之前最好对shared_ptr有所了解.如果你还不知道shared_ptr是何物,可以看看我的另 ...
- 详解C++11智能指针
前言 C++里面的四个智能指针: auto_ptr, unique_ptr,shared_ptr, weak_ptr 其中后三个是C++11支持,并且第一个已经被C++11弃用. C++11智能指针介 ...
- C++11智能指针
今晚跟同学谈了一下智能指针,突然想要看一下C++11的智能指针的实现,因此下了这篇博文. 以下代码出自于VS2012 <memory> template<class _Ty> ...
- C++11智能指针的深度理解
平时习惯使用cocos2d-x的Ref内存模式,回过头来在控制台项目中觉得c++的智能指针有点生疏,于是便重温一下.首先有请c++智能指针们登场: std::auto_ptr.std::unique_ ...
- C++11智能指针 share_ptr,unique_ptr,weak_ptr用法
0x01 智能指针简介 所谓智能指针(smart pointer)就是智能/自动化的管理指针所指向的动态资源的释放.它是存储指向动态分配(堆)对象指针的类,用于生存期控制,能够确保自动正确的销毁动 ...
- C++11智能指针原理和实现
一.智能指针起因 在C++中,动态内存的管理是由程序员自己申请和释放的,用一对运算符完成:new和delete. new:在动态内存中为对象分配一块空间并返回一个指向该对象的指针: delete:指向 ...
随机推荐
- phpcms v9联动菜单的调用方法及get_linkage函数简单过程
{get_linkage($r['areaid'],1,' >> ',1)} 复制代码 当然这个 $r 也不是绝对的.要看你的loop 是如何写的. {loop $data $n $r} ...
- CSU 1660 K-Cycle(dfs判断无向图中是否存在长度为K的环)
题意:给你一个无向图,判断是否存在长度为K的环. 思路:dfs遍历以每一个点为起点是否存在长度为k的环.dfs(now,last,step)中的now表示当前点,last表示上一个访问的 点,step ...
- 27-React Lists and Keys
Lists and Keys React支持以数组的形式来渲染多个组件,它会将你数组中的每个组件以列表的形式渲染开来. 当你使用数组的方式来渲染你的组件时,你需要给每个组件一个Key值,否则会出现一个 ...
- HashMap原理与优化
参考文献: HashMap的工作原理 java中HashMap重要性质和优化总结 一.HashMap的基本了解 基本定义:根据源代码的描述可知,HashMap是基于哈希表的Map接口的实现,其包含了M ...
- linux下配置安装python3
一.首先,官网下载python3的所需版本. wget https://www.python.org/downloads/release/python-360/Python-3.6.0.tgz 想下载 ...
- SpringMVC生成Excel下载
SpringMVC controller里的方法: @RequestMapping(value="/notify/download",produces = {"appli ...
- Mac 安装 MySQL
在 Mac 下用 Homebrew 安装 MySQL, 网上的教程倒是很多,不过大多数都很默契地雷同.如果稍有点定制要求,就无从下手了. 我先也不免俗,从基本的开始: 一.首先安装 Homebrew ...
- Strus2学习:基础(一)
Strus2基础: Sturs2起源以及背景: 在起源很早(2002年左右)的 strus1 和 webWork 基础上进行扩展,并且兼容这两大框架!总之很好用啦,随着学习的深入,应该会有更好的诠释的 ...
- Http referer origin
为了防止CSRF的攻击,我们建议修改浏览器在发送POST请求的时候加上一个Origin字段,这个Origin字段主要是用来标识出最初请求是从哪里发起的.如果浏览器不能确定源在哪里,那么在发送的请求里面 ...
- Java中如何获取spring中配置文件.properties中属性值
通过spring配置properties文件 1 2 3 4 5 6 7 8 9 <bean id="propertyConfigurer" class="co ...