你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

c++11引入了三种智能指针

[复制链接]
gaosmile 发布时间:2020-7-6 22:48
很多人谈到c++,说它特别难,可能有一部分就是因为c++的内存管理吧,不像java那样有虚拟机动态的管理内存,在程序运行过程中可能就会出现内存泄漏,然而这种问题其实都可以通过c++11引入的智能指针来解决,相反我还认为这种内存管理还是c++语言的优势,因为尽在掌握。
c++11引入了三种智能指针:
  • std::shared_ptr
  • std::weak_ptr
  • std::unique_ptr
    ; D% g& {, u) Y1 `% E9 ^4 Y
shared_ptr
shared_ptr使用了引用计数,每一个shared_ptr的拷贝都指向相同的内存,每次拷贝都会触发引用计数+1,每次生命周期结束析构的时候引用计数-1,在最后一个shared_ptr析构的时候,内存才会释放。
使用方法如下:6 }: w0 {8 S7 E
    ' m0 k* J+ c4 f; e
    1. <font style="background-color:rgb(255, 255, 255)"><font face="Tahoma"><font color="black">9 P$ y% N1 [$ ^( I& G8 L" F
    2. struct ClassWrapper {
        C+ |$ ]. t, [+ b# _+ ^# Q
    3.     ClassWrapper() {4 I; T# v, x: M
    4.         cout << "construct" << endl;
      5 I' N5 Q+ c4 q$ J! `' o* L
    5.         data = new int[10];' i4 u0 m& B" t) L5 O3 B
    6.     }
      ( x# Z* C3 Z! \' c
    7.     ~ClassWrapper() {
      $ a! M: u" g( j7 s  l7 ^- Q) W" y+ C
    8.         cout << "deconstruct" << endl;+ y, x' k! x' L6 ~; ]1 X
    9.         if (data != nullptr) {; @- a# F, `4 O
    10.             delete[] data;' X+ r$ f2 S/ @' y# z; `
    11.         }  p' h' f# K; u# l, n
    12.     }. {$ k' |! Q6 s
    13.     void Print() {
      $ N6 z! M# x: N+ _
    14.         cout << "print" << endl;8 [# Z. a! ?. M! ^: s6 G  [
    15.     }$ d* b9 `( {% o! `; k+ @$ _
    16.     int* data;
      ; ^, b6 M5 w1 `0 }: N0 S+ V$ m
    17. };
      5 \; n/ o0 ?8 Z  I$ g/ O

    18. % {  W  V! q7 @$ `; n3 C6 l
    19. void Func(std::shared_ptr<ClassWrapper> ptr) {9 I8 |* c4 D0 Z2 E8 a3 q9 a7 H  {
    20.     ptr->Print();
      " @5 P. H/ p1 N" q/ g& c0 j/ E
    21. }
      7 o, j7 r) o: g( |0 `' @( G. h) u
    22. 4 D4 [: Y8 i- O! r: j" G4 n3 ^
    23. int main() {
      / `/ h/ \# Q6 L( E4 Q, F* _
    24.     auto smart_ptr = std::make_shared<ClassWrapper>();0 z' L! _. n4 X
    25.     auto ptr2 = smart_ptr; // 引用计数+1
      / S# x, D$ q3 m& }  y& A
    26.     ptr2->Print();+ y% X7 G# T- d8 Y
    27.     Func(smart_ptr); // 引用计数+1  U3 Z+ l! L6 }) g  _; C1 d! Q
    28.     smart_ptr->Print();
      2 U0 T9 k( @' h, v1 y$ I1 q9 \
    29.     ClassWrapper *p = smart_ptr.get(); // 可以通过get获取裸指针
      ' F; I; N5 q" x$ H6 O
    30.     p->Print();( r& Q6 R; H- y& k( k6 c! m! M
    31.     return 0;- m8 O2 }- I$ @, u% w4 M
    32. }</font></font></font>
    复制代码
  • 智能指针还可以自定义删除器,在引用计数为0的时候自动调用删除器来释放对象的内存,代码如下:( ?7 L1 p; ]8 l% g
  • std::shared_ptr<int> ptr(new int, [](int *p){ delete p; });
    & ]8 z2 q4 M6 V4 x. g: t
关于shared_ptr有几点需要注意:
$ @0 ]5 E/ Z& V- T) F" g( ~
• 不要用一个裸指针初始化多个shared_ptr,会出现double_free导致程序崩溃
• 通过shared_from_this()返回this指针,不要把this指针作为shared_ptr返回出来,因为this指针本质就是裸指针,通过this返回可能 会导致重复析构,不能把this指针交给智能指针管理。
  1. $ S: ]4 Q0 r' l) N5 h% X% j4 H
  2. class A {/ C; ^, V+ R; x3 n; W
  3.     shared_ptr<A> GetSelf() {
    : D( k( N, p. R" q9 \7 H9 p
  4.        return shared_from_this();
    6 t6 m  O+ q; k5 d
  5.        // return shared_ptr<A>(this); 错误,会导致double free  J: ^0 b/ _9 Y7 b. z3 e& X
  6.     }  ' Y& ^( o  f3 q4 S
  7. };
复制代码

: B6 p& k2 ~( i  o: ?1 T* C
  • 尽量使用make_shared,少用new。
  • 不要delete get()返回来的裸指针。
  • 不是new出来的空间要自定义删除器。
  • 要避免循环引用,循环引用导致内存永远不会被释放,造成内存泄漏。
    1. ! g" q0 [" Y" C. D$ x3 b5 P
    2. using namespace std;
      , @& u: _7 `" Y! Z0 W! a
    3. struct A;4 E/ m& ?- q1 K6 Y0 [+ F4 x0 q
    4. struct B;, a: h6 u! E  {! Z& V  p" S9 r2 g

    5. $ O% \# s& _. }! |& \; @
    6. struct A {
      ) p; Y# N7 n  P/ I' a, H. s$ e. b
    7.    std::shared_ptr<B> bptr;
      & _3 m$ m2 g& L/ T0 T1 }- H; b! e! G
    8.    ~A() {
      7 h# B# M$ n9 g# ]: ^3 z
    9.        cout << "A delete" << endl;
      . _6 q$ `% u# _4 q
    10.   }. p6 V; b  r  p  m( r0 r: O+ {
    11. };
      1 A' X. L4 {: K
    12. 5 p) D2 n2 P+ D9 A0 T% ]
    13. struct B {7 ?* d1 [" H1 C+ k! @" \
    14.    std::shared_ptr<A> aptr;4 Z6 O4 L, H+ i2 U
    15.    ~B() {
      6 m0 F9 `2 {1 s5 _6 x( i' B
    16.        cout << "B delete" << endl;+ z1 A9 _: t6 `
    17.    }
      * g4 A0 Z7 i' }+ |" k) U/ R/ L+ v) w
    18. };
      ! R) j5 p! P. P

    19. 5 e5 w! j% C7 O$ T! ]
    20. int main() {
      7 Q4 e' z2 I) j# G& S
    21.    auto aaptr = std::make_shared<A>();
      9 }* I+ r9 n9 n+ m
    22.    auto bbptr = std::make_shared<B>();2 @- c; w" P6 x) m# f7 Q
    23.    aaptr->bptr = bbptr;
      7 r3 p# U9 z5 n/ k6 E7 y3 d% d
    24.    bbptr->aptr = aaptr;. I9 U2 m* ^' T: p  [; o1 W; v+ W
    25.    return 0;  A* H; s7 C; J, z9 Y! b8 m
    26. }
    复制代码
    $ d1 d# a. T% [9 M% t5 ]; N
上面代码,产生了循环引用,导致aptr和bptr的引用计数为2,离开作用域后aptr和bptr的引用计数-1,但是永远不会为0,导致指针永远不会析构,产生了内存泄漏,如何解决这种问题呢,答案是使用weak_ptr。
weak_ptr
weak_ptr是用来监视shared_ptr的生命周期,它不管理shared_ptr内部的指针,它的拷贝的析构都不会影响引用计数,纯粹是作为一个旁观者监视shared_ptr中管理的资源是否存在,可以用来返回this指针和解决循环引用问题。
  • 作用1:返回this指针,上面介绍的shared_from_this()其实就是通过weak_ptr返回的this指针,这里参考我之前写的源码分析shared_ptr实现的文章,最后附上链接。
  • 作用2:解决循环引用问题。

    1. % K, \" E/ W, o, Y# y" e- Z
    2. struct A;
      - e& T$ |6 V8 }3 u
    3. struct B;
      1 u+ {8 O, i+ c

    4. ; K1 o- n/ `9 ]/ L# @" g
    5. struct A {$ G7 g+ \. ?8 i3 {' m+ t
    6.    std::shared_ptr<B> bptr;8 ^" J: z# a, a9 g5 y9 V& C) B
    7.    ~A() {
      1 s+ f" {; s6 ]2 |! m
    8.        cout << "A delete" << endl;, f( X1 H1 m" }* p0 D
    9.    }1 i6 ]! ^, N/ s+ ]/ Y2 W
    10.    void Print() {
      6 Y2 U; g. k9 p9 Y# u! A# E* \3 l' ^
    11.        cout << "A" << endl;9 C! u9 g0 Z/ ^0 Z1 A% m) R# T
    12.    }
      $ i  L! O' Z) P5 t& |1 E
    13. };( F5 `; P* e3 ?
    14. 6 R3 X% q$ e8 Q# V
    15. struct B {7 J6 Q1 U; G: x% \1 v4 _2 l
    16.    std::weak_ptr<A> aptr; // 这里改成weak_ptr
      ( P! Y3 Q* l5 p; n) @* {! u
    17.    ~B() {
      " c3 |- O$ i, g% p
    18.        cout << "B delete" << endl;
      , V2 P  C0 A" m9 T* ^7 E0 o
    19.    }6 w' e" N% p1 o- n7 M0 i) z
    20.    void PrintA() {1 T0 D, Q: T. v% m+ j( C! R' T
    21.        if (!aptr.expired()) { // 监视shared_ptr的生命周期
      # o* M) d5 X; X" Y1 p
    22.            auto ptr = aptr.lock();) D6 a$ M& c9 G2 g% @
    23.            ptr->Print();* E) u! @; I  N" O) S: q! ^9 q
    24.       }" [- B/ L5 q  O& o
    25.    }. b- L0 r+ Z! K+ u5 e! `
    26. };" v; U1 I3 W8 |" n' Y  P
    27. ! L+ N& s* V7 s" t+ D
    28. int main() {
      ! j# S# o1 e5 s- L' E0 W
    29.    auto aaptr = std::make_shared<A>();8 |: x' ^0 T* X. a
    30.    auto bbptr = std::make_shared<B>();
      6 e* R( z6 r- w; P/ u
    31.    aaptr->bptr = bbptr;, Q7 K2 r7 n; D" ~# {
    32.    bbptr->aptr = aaptr;
      ; W; c* D' [. P" s- f4 j7 y
    33.    bbptr->PrintA();
      ' S' K- j3 o; I' c3 S0 |
    34.    return 0;3 h  w5 [* d) `" t2 {! m  n  P
    35. }
        n7 U' m% ^, S! m, R
    36. 输出:+ S8 g( J$ s0 T. W6 n* G
    37. A: X8 d2 S4 s  _9 v+ I7 A, b0 o: E
    38. A delete& w" @7 B. M2 c5 ]9 X* A: d( e; i
    39. B delete
    复制代码

    : a4 b$ \* \. x# n' I( X' J) J+ }- @
unique_ptr
std::unique_ptr是一个独占型的智能指针,它不允许其它智能指针共享其内部指针,也不允许unique_ptr的拷贝和赋值。使用方法和shared_ptr类似,区别是不可以拷贝:
  1. $ B( w! `$ v( {, q0 z" A: j' ~
  2. using namespace std;/ Y  U! `- _4 \$ z
  3. + Z1 m) o# K/ [* T8 m
  4. struct A {
    4 g/ f6 K/ K. V
  5.    ~A() {
    & u! s4 p* L1 F# v. R( ]6 C
  6.        cout << "A delete" << endl;
    5 J) q" W( o  A3 ?, j5 o
  7.    }1 G. \% M' l$ I6 [: A, _4 [, B
  8.    void Print() {! r" G: h5 J* z( h) {3 j6 p+ x
  9.        cout << "A" << endl;
    6 c, P, H4 |5 H/ E5 Q. o
  10.    }
    / d; l. M( U# I2 M- t- A
  11. };- }7 @7 Z4 @6 y2 Y" p
  12. / `  U- E  t- d# N) l
  13. 1 s2 t( t$ o  P) \& Z, m% ?
  14. int main() {+ G! R! ^. @$ V
  15.    auto ptr = std::unique_ptr<A>(new A);9 @& w& ^) c% P" q0 A" E
  16.    auto tptr = std::make_unique<A>(); // error, c++11还不行,需要c++143 C% C% P6 k: h
  17.    std::unique_ptr<A> tem = ptr; // error, unique_ptr不允许移动9 @! `  g0 C4 p8 q
  18.    ptr->Print();
    ' o! H, J: u7 k6 T
  19.    return 0;
    " c$ u5 l$ _3 e) ?- o* E. s, }* J
  20. }
复制代码

0 |: b4 M; ?  A: Y: ?# `
unique_ptr也可以像shared_ptr一样自定义删除器,使用方法和shared_ptr相同。
关于c++11的智能指针的使用就介绍到这里
- R* O, h1 B' u$ N/ f' Q# z/ H: J- z
收藏 评论0 发布时间:2020-7-6 22:48

举报

0个回答

所属标签

相似分享

关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版