class with pointer menbers

string_test.cpp

 1 #include "string.h"
2 #include <iostream>
3
4 using namespace std;
5
6 int main()
7 {
8 String s1("hello");
9 String s2("world");
10
11 String s3(s2);
12 cout << s3 << endl;
13
14 s3 = s1;
15 cout << s3 << endl;
16 cout << s2 << endl;
17 cout << s1 << endl;
18 }

string.h

 1 #ifndef __MYSTRING__
2 #define __MYSTRING__
3
4 class String
5 {
6 public:
7 String(const char* cstr=0);
8 String(const String& str);
9 String& operator=(const String& str);
10 ~String();
11 char* get_c_str() const { return m_data; }
12 private:
13 char* m_data;
14 };
15
16 #include <cstring>
17
18 inline
19 String::String(const char* cstr)
20 {
21 if (cstr) {
22 m_data = new char[strlen(cstr)+1];
23 strcpy(m_data, cstr);
24 }
25 else {
26 m_data = new char[1];
27 *m_data = '\0';
28 }
29 }
30
31 inline
32 String::~String()
33 {
34 delete[] m_data;
35 }
36
37 inline
38 String& String::operator=(const String& str)
39 {
40 if (this == &str)
41 return *this;
42
43 delete[] m_data;
44 m_data = new char[ strlen(str.m_data) + 1 ];
45 strcpy(m_data, str.m_data);
46 return *this;
47 }
48
49 inline
50 String::String(const String& str)
51 {
52 m_data = new char[ strlen(str.m_data) + 1 ];
53 strcpy(m_data, str.m_data);
54 }
55
56 #include <iostream>
57 using namespace std;
58
59 ostream& operator<<(ostream& os, const String& str)
60 {
61 os << str.get_c_str();
62 return os;
63 }
64
65 #endif

注意的点

  • string_test.cpp:11拷贝构造,14拷贝赋值
  • string.h:8拷贝构造(构造函数接受自己),9拷贝赋值,10析构函数
  • 不知道将来要创建的对象多大,所以只放一根指针,再动态创建空间放对象
  • 类中带指针,要关注三个特殊函数(Big Three)
  • 结束符号“\0”判断字符串结束
  • 21判断是否空指针
  • 22:new:,动态分配一块内存
  • 31-35:delete,析构函数,防止内存泄露,变量离开作用域时自动调用
  • 11的get_c_str()是为了配合输出函数

拷贝构造(copy ctor)

  • String a("Hello"); String b("World"); b = a;
  • 如果没有构造拷贝,会发生b和a的指针都指向“Hello”,“World”没有指针指向,导致内存泄露
  • 而且a、b一个改动会影响另一个,别名在编程中是危险的事
  • 这种拷贝方式称为“浅拷贝”,是编译器的默认版本
  • copy ctor 实现“深拷贝”,创建足够的新空间存放蓝本
  • String s2(s1); 与 String s2 = s1; 效果相同

拷贝赋值(copy op =)

  • String s1("hello"); String s2(s1); s2 = s1;
  • 过程:43删除左值--44开辟新空间--45拷贝右值
  • 40-41:检测自我赋值(self assignment),来源端和目的端是否相同
  • 这步的意义不只是提高效率,不这样写会出错
  • 因为43会把原空间删掉,导致45行复制的时候访问空指针

栈(stack)和堆(heap)

  • {Complex c1(1,2);}
  • {Complex* p = new Complex(3);
  • delete p;}
  • stack:存在于某作用域(scope)的一块内存空间,调用函数时,函数本身即形成一个stack用来防止它接受的参数,以及返回地址,离开作用域后,析构函数被自动调用(aoto object)
  • heap:system heap,操作系统提供的一块global内存空间,程序可以动态分配(dynamic allocated)从中获得若干区域,使用完后需手动释放空间
  • {static Complex c1(1,2);}
  • 静态对象,离开作用域后对象仍存在,组用域是整个程序
  • 全局对象,写在任何作用域之外(全局作用域之中),也可看做一种静态对象
  • new:先分配memory,再调用ctor,分解为以下三个函数
  • void* mem = operator new(sizeof(Complex)); // 分配内存,operator new调用malloc(n)
  • p = static_cast<Complex*>(mem); // 转型
  • p->Complex::Complex(1,2); // 构造函数,Complex::Complex(pc,1,2);
  • delete:先调用析构函数,再释放内存
  • String::~String(p); // 把字符串里面动态分配的内存删掉(字符串本身只是指针m_data)
  • operator delete(p); // 内部调用free(p),删掉字符串本身的指针

new到底分配多少内存

  • 调试模式下加上debug header 32byte,正常模式下中间为数据(一根指针4byte),上下为cookies 2*4byte,凑够16的倍数
  • 动态分配所得的array
  • new []:array new
  • delete[]:array delete
  • array new 一定要搭配 array delete
  • String* p = new String[3]; delete[] p; // 唤起3次dtor
  • String* p = new String[3]; delete p; // 唤起1次dtor,另外2个指针指向的动态内存数据无法删除(注意指针是可以删除的)
  • 虽然new complex的时候不会产生此问题,但也要注意搭配使用,养成好习惯

参考:

const在函数前面与后面的区别

https://blog.csdn.net/qq_25800311/article/details/83054129

[c++] 面向对象课程(二)-- 带指针类的设计的更多相关文章

  1. C++之不带指针类的设计——Boolean

    经典的类设计分类 带指针类 不带指针类 Header文件的布局 #ifndef __COMPLEX__ #define __COMPLEX__ #include <iostream.h> ...

  2. python笔记20(面向对象课程二)

    今日内容 类成员 成员修饰符 内容回顾 & 补充 三大特性 封装 函数封装到类 数据封装到对象 * class Foo: def __init__(self,name,age): self.n ...

  3. java 面向对象(一):类与对象

    1.面向对象学习的三条主线: * 1.Java类及类的成员:属性.方法.构造器:代码块.内部类 * * 2.面向对象的大特征:封装性.继承性.多态性.(抽象性) * * 3.其它关键字:this.su ...

  4. Lua面向对象之二:类继承

    1.类继承 ①代码 Sharp = { } --① 父类 function Sharp:new() local new_sharp = { } self.__index = self --②,self ...

  5. Expo大作战(二十七)--expo sdk api之Util(expo自带工具类),tackSnapshotAsync,Svg,SQLite

    简要:本系列文章讲会对expo进行全面的介绍,本人从2017年6月份接触expo以来,对expo的研究断断续续,一路走来将近10个月,废话不多说,接下来你看到内容,讲全部来与官网 我猜去全部机翻+个人 ...

  6. 【视频开发】OpenCV中Mat,图像二维指针和CxImage类的转换

    在做图像处理中,常用的函数接口有OpenCV中的Mat图像类,有时候需要直接用二维指针开辟内存直接存储图像数据,有时候需要用到CxImage类存储图像.本文主要是总结下这三类存储方式之间的图像数据的转 ...

  7. php课程 12-38 php的类的构造方法和析构方法怎么写

    php课程 12-38 php的类的构造方法和析构方法怎么写 一.总结 一句话总结:a.__construct(参数){},__destruct(){},b.如果类中的一个方法和类名相同,则该方法为构 ...

  8. Python 基础 面向对象之二 三大特性

    Python 基础 面向对象之二 三大特性 上一篇主要介绍了Python中,面向对象的类和对象的定义及实例的简单应用,本篇继续接着上篇来谈,在这一篇中我们重点要谈及的内容有:Python 类的成员.成 ...

  9. solr与.net系列课程(二)solr的配置文件及其含义

    solr与.net系列课程(二)solr的配置文件及其含义  本节内容还是不会涉及到.net与数据库的内容,但是不要着急,这都是学时solr必学要掌握的东西,solr可不是像其他的dll文件一样,只需 ...

随机推荐

  1. 【NX二次开发】NX对象类型及基本操作

    说明:NX中的所有对象都是通过唯一的tag_t值进行标识的,这些对象大致可以分为部件对象.UF对象.表达式.链表对象和属性对象等. 部件对象的操作: 基本操作函数: 1. UF_PART_new()  ...

  2. MySQL:如何使用MyCAT实现分库分表?

    分库分表介绍 随着微服务这种架构的兴起,我们应用从一个完整的大的应用,切分为很多可以独立提供服务的小应用.每个应用都有独立的数据库. 数据的切分分为两种: 垂直切分:按照业务模块进行切分,将不同模块的 ...

  3. Scala语言笔记 - 第二篇

    目录 1 Map的基础操作 2 Map生成view和transform解析 ​ 最近研究了下scala语言,这个语言最强大的就是它强大的函数式编程(Function Programming)能力,记录 ...

  4. ceph-csi源码分析(5)-rbd driver-nodeserver分析(上)

    更多 ceph-csi 其他源码分析,请查看下面这篇博文:kubernetes ceph-csi分析目录导航 ceph-csi源码分析(5)-rbd driver-nodeserver分析(上) 当c ...

  5. 12、elk的使用(2)

    12.8.收集日志: 因为logstash安装在从节点上,所以这里收集的主要是从节点上的服务日志: 1.收集系统日志: (1)配置文件: vim /etc/logstash/conf.d/system ...

  6. 95、配置ntp服务器

    95.1.ntp简介: ntp服务使用的是udp的123端口,如果开启了防火墙要记得放开这个端口: NTP(Network Time Protocol,网络时间协议)是用来使网络中的各个计算机时间同步 ...

  7. 【PC桌面软件的末日,手机移动端App称王】写在windows11支持安卓,macOS支持ios,龙芯支持x86和arm指令翻译

    面对这场突如其来的变革,作为软件开发者,应该如何选择自己今后的发展方向?桌面软件开发领域还有前景吗? 起源 自从苹果发布m1处理器,让自家Mac支持IOS移动端app运行之后,彻底打破了移动端app和 ...

  8. 面试题二:JVM

    JVM垃圾回收的时候如何确定垃圾? 有2种方式: 引用计数 每个对象都有一个引用计数属性,新增一个引用时计数加1,引用释放时计数减1,计数为0时可以回收: 缺点:无法解决对象循环引用的问题: 可达性分 ...

  9. RabbitMQ交换机

    RabbitMQ中,生产者并不是直接将消息发送给queue,而是先将消息发送给exchange,再由exchange通过不同的路由规则将消息路由到绑定的队列中进行存储,那么为什么要先将消息发送给exc ...

  10. 暑假自学java第二天

    今天学习了一些java规则 一个java源文件的公开类只能有一个,而且必学和源文件名相同. 了解到java的标识符规范,这对以后的团队协作有很大作用. 标识符规则和c++还是很相似的 java中的字面 ...