C++面向对象高级编程学习笔记(三)

Classes的两个经典分类

Class without pointer member(s)

complex

Class with pointer member(s)

string


带指针的class

String class


它的Header的结构


它将实现的功能:

三种构造方式:

无初值的、有初值的 和 拷贝构造。

两种操作符重载:

“<<”(输出到“cout”) & “=”(拷贝赋值)

Big Three 三个特殊函数


四个主要函数:

1.默认用指针构造;

2.拷贝构造,接受的是自身的类型的对象;

3.拷贝赋值,只有类带有指针,一定要包含有这个函数;

4.析构函数,当以这个类创建的对象,即将死亡的时候,它就会被调用。

其中2,3,4 被称为 Big Three .

ctor 和 dtor (构造函数 和 析构函数)


构造函数 ctor :

字符串会以头指针的形式传入,并以‘\0’为结尾;因此构造函数要适应不同长度的,所以用‘new’分配一块大小适合内存。

析构函数 dtor:

用于清理,清理动态分配的内存,不然就是内存泄漏了,用 delete[] 关键字释放动态分配的内存;

在调用的示例中:

指针p指向动态分配的内存,所以用 delete 释放内存;

当离开作用域的时候,会调用3次析构函数,包括以new动态分配的String对象的析构函数。

class with pointer members (带有指针的类)必须有 copy ctor (拷贝构造函数) 和 copy operator= (拷贝赋值)

如果不写拷贝赋值函数,而是直接使用原来‘=’把a的地址复制到b里去,b只会指向a所指的内存,原来所指的内存也将会丢失(造成内存泄漏),这叫浅拷贝,不是我们想要的结果,我们要的是把内容拷贝过来的深拷贝。

在设计中应该避免有别名出现,即多个指针指向同一块内存。

copy ctor (拷贝构造函数)

copy assignment operator(拷贝赋值函数)

经典的写法

① 释放原来的内存

② 重新申请一个足够大的内存

③ 将内容拷贝进来

然后,在最前面检测自我赋值的情况;如果出现自我赋值,而又没有检测自我赋值,这样会因为内存被释放而丢失原来的内容。

(这里的this是指向调用者的,而且这个函数它还是成员函数,所以可以直接更改)

output函数

operator<< 函数一定是全局函数,如果它是成员函数的话cout就要在右边,因为成员函数的左边一定是自身类的指针 *this

所谓stack(栈) ,所谓heap(堆)

Stack,是存在于作用域(scope)的一块内存空间(memory space)。例如当你调用函数,函数本身即会形成一个stack用来旋转它所接收的参数,以及返回地址。

在函数本体(function body)内声明的任何变量,其所使用的内存块都取自上述stack。

Heap,或调system heap,是指由操作系统提供的一块global(全局的)内存空间,程序可动态分配,从中获得若干块。

c1所占用的空间来自Stack;当离开作用域时,它的生命自然结束;

Complex(3)是个临时创建的对象,占用的空间是以new动态分配非得,并由heap提供;它必须需要手动delete掉。

关于生命期

stack object:

c1便是,其生命在作用域结束之际结束。又称为auto object,因为它会被[自动]清理。

static local object(静态对象):

c2便是所谓static object,其生命在作用域结束之后仍然存在,直到整个程序结束。

global object(全局对象):

c3便是所谓global object,它的生命在整个程序结束之后才结束。也可以把它被视为一种static object,其作用域是[整个程序]。

heap objects:

p所指的便是heap object,其生命在它被deleted之际结束。

在创建时,我们得到的是一个指针p,所以我们应该delete指针p,同时会调用被删除对象内的析构函数。

如果没把heap对象 delete掉,就会出现内存泄漏(memory leak),即是程序失去对内存块的控制(内存块再也找不回来了)

以上便会出现内存泄漏,因为当作用域结束,p所指的heap object仍然存在,但指针p的生命却结束了,作用域之外再也看不到p(也就没机会delete p)

动态分配所得的内存块(关于new和delete)

new其实也是一个函数

new调用时,先分配memory,再调用ctor

以上便是编译器将new转化的三个动作,实际上调了malloc()函数分配内存。

delete:先调用dtor,再释放memory

编译器转化的两个动作,它里面也调用了free()函数。

动态分配数组对象

以上的用法叫做array new和array delete;

array new一定要搭配array delete,因为delete只是会删除指针所指的内存,数组内的其它内存就会因此而丢失(内存泄漏)。

(注:动态分配的内存区块中,头尾也会有一个称为cookie标识的内存块,标识中用16进位表示状态,结尾为1表示“分配”,并且动态分配的内存区块一定是16的倍数)

扩展补充

关于static

一般的成员函数都会有this point

以上是从编译器角度看到的成员函数的调用。

如果一个数据在每个同类的对象都是相同的,这时候它就是应该变成static数据,在它的声明前加上static便是;

static数据要在class body外定义,上面黄色部分便是。

static函数:它只能处理static的数据,也没this point;它可以通过object调用(通过已经创建的对象调用),也可以通过class name调用(直接修改类的static数据,作用于所有通过这个类创建的对象)

关于把ctors放在private区

之前说过这种把构造函数放在private中的类,叫做Singleton(单体)

上面的便所谓的Singleton,因为它不能被外境构造,所以它须要在自身内被构造,并且一定是static的;

因为static函数可以通过class name调用(上面的小框便是),通过getInstance()调用‘a’,这样就解决了不能调用的问题。

但就是因为static的数据一被定义了就会直到整个程序结束,它的生命才会结束;

用上面的写法可以解决static object不被使用也一直存在的问题。

关于cout

cout属于右上角的class,而这个类型继承自ostream,所以它也是属于ostream。

从ostream里可以发现,它做了好多种operator<<的重载,这正是cout可以接受多种类型数据的原因。

class template(类模板)

如果在设计没有确定类里面的参数在使用时的类型,可以通过类模板在使用时再确定它们的数据类型。

写法如上:在类的前面声明T是一个关键字;用T替换参数类型的位置。

图中左下角便是它的调用方法。

function template(函数模板)

用法:正如上面的比大小函数,为了适用于所有不同类型的对象,所以它采用了函数模板,代替未知的对象类型;它的声明和类模板相似,要在函数之前用加上黄色的语句。

(使用这种方式时,不要忘了还要对相应的操作符进行重载)

namespace

它的作用是将它里面的东西包装起来,防止和其它人重名。(std是指标准库,标准库的所有东西都被包在std里)

用法:

最简单的用法就是using directive(全开)(写法如上),这样在下面的所以标准库的函数都不用写命名了(cin和cout的全名是std::cin和std::cout)。

using declaration(逐个打开‘声明’)

正如上面的,它只声明了cout,所以就是只有cout不用写全名。

更多拓展,此课程不再详述

Composition(复合),表示has-a

表示queue(队列,先进先出)包含deque(两端都可以进出的队列);

queue拥有deque的所有功能,这样便是所谓Composition,而queue自身没有实现功能只是改变功能的名字,这类情况叫做Adapter(改装);

Adapter:适用于已经存在一个类能实现所需的功能时,只是有一些情况不同(可能是接口不一样-函数名不同),它是复合中一种特殊的情况

Composition类的大小要加上它所包含的所有类的大小

Composition关系下的构造和析构

当Container包含Component时:

编译器会在Container的构造函数名之后,自动加上Component的默认构造(‘ : Component() ’),这便使它先要执行所含类的构造函数(如果需要调用所含类的其它构造函数,需要自己写上)然后才执行自己;

编译器也会在Container的析构函数中加上所含有类的析构函数,且是先执行自己。

Delegation (委托) . Composition by reference

这里是通过指针,指向功能实现的类,而自身却只是一个对外的接口,所有功能的实现都是通过指针委托‘功能实现类’完成的,这样它便有了高度的弹性(可以改变它功能,而不影响整体,也可以方便地增加它的功能)

这样的写法称为pimpl(private implementation),左边的是Handle,右边的是Body,其主要作用是解开类的使用接口和实现的耦合

(这里用到是指针,但为什么也叫by reference而不是by point?因为业界中没有by point这个说法,by point也叫by reference)

Inheritance(继承),表示is-a

(在C++中,struct其实是一种class)

语法:就是加黄色的一行;上面的是父类,下面的是子类(子类继承父类)。

C++的继承拥有三种方式,分别是public / private / protected,其中最重要的public。(而Java中只有public)

子class拥有自己的part(成份)同时,还涵盖了父class的prat。

图示,用空心三角形表示继承。

Inheritance(继承)关系下的构造和析构

与Composition的关系相似,但前提是父类中的构造函数一定要是虚函数(后面会解释虚函数),否则不会出图中下面是两个动作

Inheritance(继承)with virtual functions(虚函数)

父类的成员函数分三种类型:非虚函数、虚函数、纯虚函数。

虚函数的语法:在非虚函数之前加virtual。

非虚函数,子类不能重新定义,在父类中定义好,供子类使用。

虚函数,在父类中定义,子类也可以重新定义它。

纯虚函数,在父类中声明,在子类中定义。

Template Method(模板方法):二十三种设计模式之一。

示例中,通过子类的对象调用父类的函数,当调用虚函数时,编译器会检查子类是否有重新定义。

Inheritance + Composition关系下的构造和析构

在第一种情况中,Base part和Component part它两的构造的先后是顺序,而析构则是逆序的。

在第二种情况中,与之前的相类似,不多说。

Delegation(委托) + Inheritance(继承)

左边委托右边,右边作为父类,可以被继承。

这样的写法可以使左边被创建,内容可以被多个继承右边的子类观察。(就像一个文件在同一个软件里被打开,但有不同的查看窗口)

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

推荐阅读更多精彩内容