c++智能指针用法

智能指针是什么

智能指针是
c++中有四个智能指针:auto_ptr、shared_ptr、weak_ptr、unique_ptr,其中后三个是c++11支持,并且第一个已经被c++11弃用。
智能指针是一个RAII(Resource Acquisition is initialization)类模型,用来动态的分配内存。当超出类的作用域时,类会自动调用析构函数,析构函数会自动释放资源。
举例:当我们写一个new语句时,一般会立即把delete语句也直接写入,但是不能避免程序还未执行到delete时就跳转或在函数没有执行到最后的delete语句就返回了,如果不在每一个可能跳转或返回的语句前释放资源,就会造成内存泄露,使用智能指针可以很大程度避免这个问题。
auto_ptr的类模板原型为:

templet<class T>
class auto_ptr {
  explicit auto_ptr(X* p = 0) ; 
  ...
};
  • 定义一个类
class Test
{
public:
    Test(string s)
    {
        str = s;
       cout<<"Test creat\n";
    }
    ~Test()
    {
        cout<<"Test delete:"<<str<<endl;
    }
    string& getStr()
    {
        return str;
    }
    void setStr(string s)
    {
        str = s;
    }
    void print()
    {
        cout<<str<<endl;
    }
private:
    string str;
};
  • auto_ptr智能指针
  1. auto_ptr访问自己的成员函数用.,访问指向对象的成员时用->
  2. 对智能指针进行赋值时,ptest2 = ptest;。此时ptest2会接管ptest原来的内存管理权,ptest会变成空指针;如果ptest2原来不为空,则它会释放原来的资源。判断一个智能指针是否为空不能用if(ptest == NULL),应该用if(ptest.get() == NULL)
  3. auto_ptr的成员函数release,作用:把智能指针赋值为空,只释放对资源的所有权,原来指向的内存并不被释放。
  4. 若中途想释放资源,可以使用ptest.reset();,就不用等到智能指针被析构时释放。
int main()
{
    auto_ptr<Test> ptest(new Test("123"));
    auto_ptr<Test> ptest2(new Test("456"));
    ptest2 = ptest;
    ptest2->print();
    if(ptest.get() == NULL)cout<<"ptest = NULL\n";
    ptest->setStr("hello ");
    ptest->print();
    ptest.get()->print();
    ptest->getStr() += "world !";
    (*ptest).print();
    ptest.reset(new Test("123")); //重新绑定指向的对象,原来的对象会被释放
    ptest->print();
    return 0;
}
  • unique_ptr智能指针
  1. 独享所有权,拥有它指向的对象
  2. 无法进行复制构造,无法使两个unique_ptr指向同一个对象。应该用std::move()
  3. 保存指向某个对象的指针,当它本身被删除释放的时候,会使用给定的删除器释放它指向的对象。
    4)可以用if(ptest == NULL)来判断是否空指针;
unique_ptr<Test> fun()
{
    return unique_ptr<Test>(new Test("789"));
}
int main()
{
    unique_ptr<Test> ptest(new Test("123"));
    unique_ptr<Test> ptest2(new Test("456"));
    ptest->print();
    ptest2 = std::move(ptest);//不能直接ptest2 = ptest
    if(ptest == NULL)cout<<"ptest = NULL\n";
    Test* p = ptest2.release();
    p->print();
    ptest.reset(p);
    ptest->print();
    ptest2 = fun(); //这里可以用=,因为使用了移动构造函数
    ptest2->print();
    return 0;
}

编译运行结果:


image.png
  • share_ptr智能指针
  1. 资源可以被多个指针共享,使用计数机制表明资源被几个指针共享。通过成员函数use_count()来查看资源的所有者个数。
  2. 调用release()时,当前指针会释放资源所有权,计数减一。当计数等于0时,资源会被释放。
int main()
{
    shared_ptr<Test> ptest(new Test("123"));
    shared_ptr<Test> ptest2(new Test("456"));
    cout<<ptest2->getStr()<<endl;
    cout<<ptest2.use_count()<<endl;
    ptest = ptest2;//"456"引用次数加1,“123”销毁
    ptest->print();
    cout<<ptest2.use_count()<<endl;//2
    cout<<ptest.use_count()<<endl;//2
    ptest.reset();
    ptest2.reset();//此时“456”销毁
    cout<<"done !\n";
    return 0;
}
image.png
  • weak_ptr智能指针
  1. 用来解决shared_ptr相互引用时的死锁问题,若两个shared_ptr相互引用,那么这两个指针的引用计数永远不可能下降为0,资源永远不会释放。
  2. 是对对象的一种弱引用,不会增加对象的引用计数,和shared_ptr之间可以相互转化,shared_ptr可以直接赋值给它,通过调用lock函数获得shared_ptr.
class B;
class A
{
public:
    shared_ptr<B> pb_;
    ~A()
    {
        cout<<"A delete\n";
    }
};
class B
{
public:
    shared_ptr<A> pa_;
    ~B()
    {
        cout<<"B delete\n";
    }
};
 
void fun()
{
    shared_ptr<B> pb(new B());
    shared_ptr<A> pa(new A());
    pb->pa_ = pa;
    pa->pb_ = pb;
    cout<<pb.use_count()<<endl;
    cout<<pa.use_count()<<endl;
}
 
int main()
{
    fun();
    return 0;
}

上述fun函数中pa、pb之间相互引用,两个资源的引用计数为2,当要跳出函数时,智能指针pa、pb析构时两个资源引用计数会减一,但是两者引用计数还是为1,导致调出函数时资源没有被释放(A、B的析构函数没有被调用)。运行结果如下:


image.png

若把类A中的shared_ptr<B> pb_改为weak_ptr<B> pb_,资源B的引用开始只有1,当pb析构时,B的计数变为0,B得到释放,B释放的同时会使A的计数减一,同时pa析构时使A的计数减一,那么A的计数为0,A得到释放。编译运行结果如下:

image.png

  1. 是对对象的一种弱引用,不会增加对象的引用计数,和shared_ptr之间可以相互转化,shared_ptr可以直接赋值给它,通过调用lock函数获得shared_ptr.不能通过weak_ptr直接访问对象。
    若B对象中有一个方法print(),不能通过如下方式访问
pa->pb_->print()

因为pb_是一个weak_ptr,应先把它转换为shared_ptr。

shared_ptr<B> p = pa->pb_.lock();
p->print();

参考大佬博客

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,544评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,430评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,764评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,193评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,216评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,182评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,063评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,917评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,329评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,543评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,722评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,425评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,019评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,671评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,825评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,729评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,614评论 2 353