1、C++中一般採用以下三种方法之中的一个管理指针成员:

(1)指针成员採取常规行为。

这种类具有指针的全部缺陷:具有指针成员且使用默认复制构造函数和赋值操作符,无法避免悬垂指针(两个对象的指针成员指向同一内存。删除了当中一个指针指向的内存时,还有一个指针将不再指向有效的内存空间)。

(2)类能够实现所谓的"智能指针"行为。引入计数类,智能指针类将一个计数器与类的对象相关联。使用计数跟踪该类有多少个对象共享同一指针。当计数为0时。删除对象。

(3)类採取值行为。採用重载的复制构造函数、赋值操作符和析构函数。

2、指针成员採取常规行为演示样例:两个指针指向同一块内存,会引起不可预料的后果

  1. #include "stdafx.h"
  2. #include <string.h>
  3. #include <iostream.h>
  4. class HasPtr
  5. {
  6. public:
  7. HasPtr(int *p,int i):ptr(p),val(i){}
  8. int *get_ptr()const{return ptr;}
  9. int get_val()const{return val;}
  10. void set_ptr(int *p){ptr=p;}
  11. void set_val(int i){val=i;}
  12. int get_ptr_val()const{return *ptr;}
  13. void set_ptr_val(int val)const{*ptr=val;}
  14. private:
  15. int *ptr;
  16. int val;
  17. };
  18. int main(int argc, char* argv[])
  19. {
  20. int *p=new int(10);
  21. cout<<p<<endl;
  22. HasPtr ptr(p,10);
  23. cout<<ptr.get_ptr()<<endl;
  24. delete p;
  25. cout<<ptr.get_ptr_val()<<endl;  //p和ptr中的指针指向同一对象。删除该对象后,ptr中的指针不再指向有效的对象。
  26. return 0;
  27. }

3、"智能指针"行为演示样例:注意构造函数

  1. #include "stdafx.h"
  2. #include <string.h>
  3. #include <iostream.h>
  4. class HasPtr;
  5. //计数类U_Ptr全部成员均为private,将HasPtr设置为计数类的友元类,使其能够訪问U_Ptr的成员
  6. class U_Ptr
  7. {
  8. friend class HasPtr;
  9. int *ip;
  10. size_t use;
  11. U_Ptr(int *p):ip(p),use(1){}
  12. ~U_Ptr()
  13. {
  14. delete ip;
  15. }
  16. };
  17. class HasPtr
  18. {
  19. public:
  20. HasPtr(int *p,int i):ptr(new U_Ptr(new int(*p))),val(i){} //构造函数,创建新的U_Ptr对象
  21. HasPtr(const HasPtr &orig):ptr(orig.ptr),val(orig.val){++ptr->use;} //复制构造函数,计数加1
  22. HasPtr& operator=(const HasPtr &rhs)  //赋值操作符,左操作数计数减1,右操作数计数加1,假设左操作数减至0,则删除左操作数指向的对象
  23. {
  24. if (this!=&rhs)
  25. {
  26. ++rhs.ptr->use;
  27. if(--ptr->use==0)
  28. delete ptr;
  29. ptr=rhs.ptr;
  30. val=rhs.val;
  31. }
  32. return *this;
  33. }
  34. ~HasPtr() //析构函数,计数减1,假设计数减至0,就删除对象
  35. {
  36. if (--ptr->use==0)
  37. {
  38. delete ptr;
  39. }
  40. }
  41. int *get_ptr()const{return ptr->ip;}
  42. int get_val()const{return val;}
  43. void set_ptr(int *p){ptr->ip=p;}
  44. void set_val(int i){val=i;}
  45. int get_ptr_val()const{return *ptr->ip;}
  46. void set_ptr_val(int val){*ptr->ip=val;}
  47. private:
  48. U_Ptr *ptr;
  49. int val;
  50. };
  51. int main(int argc, char* argv[])
  52. {
  53. int *p=new int(10);
  54. cout<<p<<endl;
  55. HasPtr ptr(p,10);
  56. cout<<ptr.get_ptr()<<endl;        //两指针指向同一块内存
  57. cout<<ptr.get_ptr_val()<<endl;
  58. delete p;
  59. return 0;
  60. }

4、定义值型类:三法则(赋值操作符、复制构造函数、析构函数)

  1. #include <string.h>
  2. #include <iostream.h>
  3. class HasPtr
  4. {
  5. public:
  6. HasPtr(int *p,int i):ptr(new int(*p)),val(i){} //构造函数
  7. HasPtr(const HasPtr &orig):ptr(new int(*orig.ptr)),val(orig.val){} //复制构造函数
  8. HasPtr& operator=(const HasPtr &rhs)  //赋值操作符
  9. {
  10. if (this!=&rhs)
  11. {
  12. ptr=new int(*rhs.ptr);
  13. val=rhs.val;
  14. }
  15. return *this;
  16. }
  17. ~HasPtr(){delete ptr;}  //析构函数
  18. int *get_ptr()const{return ptr;}
  19. int get_val()const{return val;}
  20. void set_ptr(int *p){ptr=p;}
  21. void set_val(int i){val=i;}
  22. int get_ptr_val()const{return *ptr;}
  23. void set_ptr_val(int val)const{*ptr=val;}
  24. private:
  25. int *ptr;
  26. int val;
  27. };
  28. int main(int argc, char* argv[])
  29. {
  30. int *p=new int(10);
  31. cout<<p<<endl;
  32. HasPtr ptr(p,10);
  33. cout<<ptr.get_ptr()<<endl;       //p与ptr的指针不是指在同一块内存,可是所指向的对象内容是一样的
  34. delete p;
  35. cout<<ptr.get_ptr_val()<<endl;
  36. return 0;
  37. }

C++管理指针成员的更多相关文章

  1. C++ Primer 学习笔记_57_类和数据抽象 --管理指针成员

    复印控制 --管理指针成员 引言: 包括指针的类须要特别注意复制控制.原因是复制指针时.一个带指针成员的指针类 class HasPtr { public: HasPtr(int *p,int i): ...

  2. 【c++】类管理指针成员

    c++编程提倡使用标准库,一个原因是标准库大胆减少对指针的使用.但是许多程序是离不开指针的.包含指针的类需要特别注意复制控制,原因是复制指针时只复制指针中的地址,而不复制指针所指向的对象.这样当把一个 ...

  3. C++ Primer 有感(管理类的指针成员)

    C++类的指针成员与其他成员有所不同,指针成员指向一个内存地址,该地址的内存需要我没管理. 我现在分析一下为什么要管理指针成员. 有如下Student类,Student.h如下: [cpp] view ...

  4. C++ 带有指针成员的类处理方式

    在一个类中,如果类没有指针成员,一切方便,因为默认合成的析构函数会自动处理所有的内存.但是如果一个类带了指针成员,那么需要我们自己来写一个析构函数来管理内存.在<<c++ primer&g ...

  5. 关于Vector中的元素中含有指针成员的情况

    对于容器,当容器的各个元素为类类型,且该类类型中含有指针成员时: 如果类类型的析构函数中包含了对指针变量指向内存的释放操作,则在利用clear()函数删除容器所有元素时,会自动调用类的析构函数,自动实 ...

  6. 字符串输出输入函数,const修饰符,内存分区,动态内存管理,指针和函数,结构体

    1.字符串输出输入函数 读入字符串的方法: 1) scanf 特点:不能接收空格 2) gets 特点:可以接受含有空格的字符串 ,不安全 3) fgets(); 特点:可以帮我们自动根据数组的长度截 ...

  7. C++对象的复制——具有指针成员的类的对象的复制

    //smartvessel@gmail.com class Table{ Name * p; size_t sz; publish: Table(size_t s = 15){p = new Name ...

  8. C++函数指针相关 & 类成员的指针 & 成员函数的指针

    有时候会有指向类成员变量或者成员函数的指针,但是注意,这个指针并不是针对一个地址的指向,而更多的是一个偏移. 同时,支持将父类对象的成员 转为 子类对象的成员指针,如下: 反过来,是不行的.因为父类的 ...

  9. 15.含有指针成员的类的拷贝[ClassCopyConstructorWithPointerMember]

    [题目] 下面是一个数组类的声明与实现.请分析这个类有什么问题,并针对存在的问题提出几种解决方案.  C++ Code  123456789101112131415161718192021222324 ...

随机推荐

  1. PCB MS SQL 通过表名查询各字段信息和vb.net C# module类代码

    正式表:各字段内容获取 ) SET @tabname = 'ppeflow' SELECT @tabname AS '表名' ,(CASE ))+ ')' )) ) )) + ')' )) ) )) ...

  2. selenium3 + python - page_source页面源码

    前言: 有时候通过元素的属性的查找页面上的某个元素,可能不太好找,这时候可以从源码中爬出想要的信息.selenium的page_source方法可以获取到页面源码. 本次以博客园为例,先爬取页面源码, ...

  3. input点击修改样式

    <input id="geren" type="button" value="个人奖励" style="BORDER-TOP ...

  4. 多文件上传ajax jquery

    jquery的ajaxSubmit()和多文件上传 <%@ page language="java" import="java.util.*" pageE ...

  5. unicode、UTF-8、UTF-16的历史

    1:中国人民通过对 ASCII 编码的中文扩充改造,产生了 GB2312 编码,可以表示6000多个常用汉字. 2:汉字实在是太多了,包括繁体和各种字符,于是产生了 GBK 编码,它包括了 GB231 ...

  6. SQL server基本语法

    此处源于一个基本的SQL Server试题,基本上涵盖了SQL Server的全部基本语法,粘贴在此处,权当分享   --1.  创建TestDB数据库 create database TestDB; ...

  7. 百鸡百钱===百马百担====for循环嵌套

    package com.zuoye.test;//百鸡百钱5文钱可以买一只公鸡,3文钱可以买一只母鸡,1文钱可以买3只雏鸡.public class Baiji { public static voi ...

  8. 通过Static 字段来维护状态是不是一个好主意

    static是申明静态字段.静态方法或者静态类的修饰符.使用static申明的字段属于类型本身而不属于任何字段,声明的类也具有一些特别特性,比如不能实例化,不能继承等.用通俗化的语言来说,static ...

  9. (转)基于Metronic的Bootstrap开发框架经验总结(2)--列表分页处理和插件JSTree的使用

    http://www.cnblogs.com/wuhuacong/p/4759564.html 在上篇<基于Metronic的Bootstrap开发框架经验总结(1)-框架总览及菜单模块的处理& ...

  10. java操作Excel的poi的字体设置

    package com.java.poi; import org.apache.poi.hssf.usermodel.HSSFWorkbook;import org.apache.poi.ss.use ...