C++ Primer 有感(管理类的指针成员)
C++类的指针成员与其他成员有所不同,指针成员指向一个内存地址,该地址的内存需要我没管理。
我现在分析一下为什么要管理指针成员。
有如下Student类,Student.h如下:
- class Student
- {
- public:
- Student(int *books);
- virtual ~Student();
- int *books;
- };
Student.cpp如下:
- #include "Student.h"
- Student::Student(int *books)
- {
- this->books=books;
- }
- Student::~Student()
- {
- }
在主函数中我如下写:
- #include <iostream>
- #include "Student.h"
- using namespace std;
- int main()
- {
- int *b=new int(34);
- Student s(b);
- delete b;
- cout<<(*(s.books))<<endl;
- return 0;
- }
当我释放掉了b所指的空间,b的地址就不能用了,但是在s中任然可以访问,所以就出现了不可预知的错误
最后输出的结果如下:

这个指针所指的空间需要我们的Student类来管理,因为我们不知道什么时候该释放掉b的空间,如果手动释放掉b的空间,然后再使用Student对象引用该空间的值,就会出现上面的错误。更复杂的情况是使用一个Student对象初始化另外一个Student对象的时候和赋值的时候。
管理指针成员有两种办法
1.定义智能指针类
2.定义值类型
下面分别介绍
1.智能指针
智能指针:一个行为类似指针但也提供其他功能的类。智能指针的一个通用形式接受指向动态分配对象的指针并负责删除掉该对象,用户分配对象,但由智能指针类删除掉它。智能指针类需要实现复制控制成员来管理指向共享对象的指针。只有在撤销了指向共享对象的最后一个智能指针后,才能删除该共享对象。使用计数是实现智能指针类最常用的方式。
计数类的实现
我先设计一个数据类,目的是察看智能指针的效果。数据类就一个数据,int类型,我主要在析构函数中输出删除了该对象。Flag.h头文件:
- class Flag {
- public:
- Flag(int v);
- virtual ~Flag();
- int i;
- };
Flag.cpp源文件:
- #include "Flag.h"
- #include <iostream>
- Flag::Flag(int v) {
- // TODO Auto-generated constructor stub
- i=v;
- }
- Flag::~Flag() {
- std::cout<<"delete the value"<<std::endl;
- }
下面实现记述类:U_Ptr.h文件:
- #include "Flag.h"
- class U_Ptr {
- public:
- friend class Student;
- U_Ptr(Flag *p);
- virtual ~U_Ptr();
- private:
- Flag *books;
- unsigned int use;
- };
U_Ptr.cpp源文件:
- #include "UPtr.h"
- U_Ptr::U_Ptr(Flag *p) {
- this->books=p;
- this->use=1;
- }
- U_Ptr::~U_Ptr() {
- delete books;
- }
这个类设置Student类为它的友元,以便在Student类中访问该类的私有成员。该类接收一个Flag的指针,该内存在其他地方分配,由U_Ptr类来管理,当该类被释放的时候回收Flag指针所指内存。
下面我完成智能指针类Student,该类决定在什么样的情况下删除Flag指针所分配的内存。
Student.h头文件:
- #include "UPtr.h"
- #include "Flag.h"
- class Student {
- public:
- Student(Flag *p);
- Student(const Student &s);
- Student& operator=(const Student &s);
- int getValue();
- void setValue(int number);
- virtual ~Student();
- private:
- U_Ptr *ptr;
- };
该类包含了一个U_Ptr对象的指针,在这个类中,要对复制与赋值进行控制。
Student.cpp源文件:
- #include "Student.h"
- Student::Student(Flag *p) {
- this->ptr=new U_Ptr(p);
- }
- Student::Student(const Student &s)
- {
- this->ptr=s.ptr;
- ++ptr->use;
- }
- Student& Student::operator=(const Student &s)
- {
- ++s.ptr->use;
- if(--ptr->use==0)
- {
- delete ptr;
- }
- ptr=s.ptr;
- return *this;
- }
- int Student::getValue()
- {
- return (ptr->books)->i;
- }
- void Student::setValue(int number)
- {
- (ptr->books)->i=number;
- }
- Student::~Student() {
- if(--ptr->use==0)
- {
- delete ptr;
- }
- }
在构造函数中,为ptr指针分配内存,但是这个内存在什么时候释放掉要看该ptr的use的计数,当use为0的时候,说明没有其他的Student类共享该ptr所指内存,这时就释放内存。
在赋值的时候,先将右值的计数加一,本对象的计数应该减一,并且判断这时是否应该释放该对象,然后将右值的ptr赋值给本对象。这样如果是相同的对象赋值的话,有安全的保障。如果先减一,刚好计数为1,会释放内存,然后将计数加一。
在主函数中执行下面的代码会得到我希望的结果:
- #include <iostream>
- using namespace std;
- #include "Flag.h"
- #include "Student.h"
- int main() {
- Flag *f=new Flag(43);
- Student s1(f);
- Student s2(s1);
- return 0;
- }
打印出 delete the value.
2.定义值类型
这种方法比较简单,看代码了:
Human.h类就是这样的类:
Human.h
- class Human {
- public:
- Human(int value);
- Human(const Human &value);
- Human& operator=(const Human &value);
- virtual ~Human();
- private:
- int *ip;
- };
Human.cpp
- #include "Human.h"
- Human::Human(int value) {
- // TODO Auto-generated constructor stub
- ip=new int(value);
- }
- Human::Human(const Human &value) {
- // TODO Auto-generated constructor stub
- ip=new int(*(value.ip));
- }
- Human& Human::operator=(const Human& value)
- {
- *ip=*(value.ip);
- return *this;
- }
- Human::~Human() {
- // TODO Auto-generated destructor stub
- delete ip;
- }
在构造函数中为指针分配内存,在析构函数中释放内存,在赋值操作中改变指针所指内容的值而不是指针的值。
C++ Primer 有感(管理类的指针成员)的更多相关文章
- C++ Primer 有感(类)
1.在类内部,声明成员函数时必需 的,而定义成员函数则是可选的.在类内部定义的函数默认为inline. 2.const成员函数不能改变其所操作的对象的数据成员.const必须同时出现在声明和定义中,若 ...
- C++ Primer 学习笔记_57_类和数据抽象 --管理指针成员
复印控制 --管理指针成员 引言: 包括指针的类须要特别注意复制控制.原因是复制指针时.一个带指针成员的指针类 class HasPtr { public: HasPtr(int *p,int i): ...
- 【c++】类管理指针成员
c++编程提倡使用标准库,一个原因是标准库大胆减少对指针的使用.但是许多程序是离不开指针的.包含指针的类需要特别注意复制控制,原因是复制指针时只复制指针中的地址,而不复制指针所指向的对象.这样当把一个 ...
- C++ 带有指针成员的类处理方式
在一个类中,如果类没有指针成员,一切方便,因为默认合成的析构函数会自动处理所有的内存.但是如果一个类带了指针成员,那么需要我们自己来写一个析构函数来管理内存.在<<c++ primer&g ...
- C++智能指针管理类
1.程序员明确的进行内存释放 对于c++程序员,最头脑的莫过于对动态分配的内存进行管理了.c++在堆上分配的内存,需要程序员负责对分配的内存进行释放.但有时内存的释放看起来并不件很轻松的事,如下程序 ...
- 类1(this指针/const成员函数/类作用域/外部成员函数/返回this对象的函数)
假设我们要设计一个包含以下操作的 Sales_data 类: 1.一个 isbn 成员函数,用于返回对象的 book_no 成员变量 2.一个 combine 成员函数,用于将一个 Sales_dat ...
- C++管理指针成员
1.C++中一般採用以下三种方法之中的一个管理指针成员: (1)指针成员採取常规行为. 这种类具有指针的全部缺陷:具有指针成员且使用默认复制构造函数和赋值操作符,无法避免悬垂指针(两个对象的指针成员指 ...
- YTU 2636: B3 指向基类的指针访问派生类的成员函数
2636: B3 指向基类的指针访问派生类的成员函数 时间限制: 1 Sec 内存限制: 128 MB 提交: 433 解决: 141 题目描述 领导类(Leader)和工程师类(Engineer ...
- 15.含有指针成员的类的拷贝[ClassCopyConstructorWithPointerMember]
[题目] 下面是一个数组类的声明与实现.请分析这个类有什么问题,并针对存在的问题提出几种解决方案. C++ Code 123456789101112131415161718192021222324 ...
随机推荐
- Android TextView常用属性
[说明] TextView是用来显示文本的组件.以下介绍的是XML代码中的属性,在java代码中同样可通过 "组件名.setXXX()方法设置.如,tv.setTextColor(); [属 ...
- ubuntu 命令行下格式化U盘,磁盘分区
命令行格式化磁盘一般是:先卸载,后格式化. 先说格式化U盘的方法,格式化磁盘某个分区是同样的道理. 一般情况下U盘会挂载在/meida/<username>/<disk>目录下 ...
- C++笔记002:VS2010报错:LINK fatal error LNK1123 转换到 COFF 期间失败文件无效或损坏
原创笔记,转载请注明出处! 点击[关注],关注也是一种美德~ 错误描述: 1>------ 已启动生成: 项目: FirstCode, 配置: Debug Win32 ------ 1>生 ...
- Node.js Query Strings
Query String 稳定性: 3 - 稳定 这个模块提供了一些处理 query strings 的工具,包括以下方法: querystring.stringify(obj[, sep][, eq ...
- Redis之(四)事务
5.1开始事务 MULTI 命令的执行标记着事务的开始: 当客户端处于非事务状态下时, 所有发送给服务器端的命令都会立即被服务器执行. Redis 的事务不可嵌套, 当客户端已经处于事务状态, 而客户 ...
- 【mybatis深度历险系列】延迟加载
在前面的博文中,小编主要简单的介绍了mybatis中的高级映射,小伙伴们可以把mybatis和hibernate的因素进行对比,更加有利于理解.今天这篇博文,小编主要来简单介绍一下mybatis中的延 ...
- Xcode Organizational Identifiers
操作系统(不管是iOS或是OS X)使用bundle标识去唯一标识你的应用.Bundle标识由一个组织id和你App的名字组成. 一般的,组织id是你域名的反转.如果你的域名是example.com那 ...
- Android TV开发总结(二)构建一个TV Metro界面(仿泰捷视频TV版)
前言:上篇是介绍构建TV app前要知道的一些事儿,开发Android TV和手机本质上没有太大的区别,屏大,焦点处理,按键处理,是有别于有手机和Pad的实质区别.今天来介绍TV中Metro UI风格 ...
- Racket 模拟SICP的流(延时计算)
默认的Racket是要对函数参数进行求值的, 例如(f 1 (+ 1 2))里面,(+ 1 2)要先求值为3,变为(f 1 3)再进行下一步操作.因此, Racket若按照SICP使用define关键 ...
- ELK平台的搭建
ELK是指Elasticsearch + Logstash + Kibaba三个组件的组合.本文讲解一个基于日志文件的ELK平台的搭建过程,有关ELK的原理以及更多其他信息,会在接下来的文章中继续研究 ...