一、让自己习惯C++

第一章介绍 C++ 的一些基本方式。

1. 视 C++ 为一个语言联邦

现在 C++ 已经是个多重泛型编程语言,同时支持过程、面向对象、函数、泛型、元编程形式。将其视为语言联邦,主要有四个次语言:

  • C
    区块、语句、预处理、内置数据类型、数组、指针都来自C。此时传值比传引用高效。
  • Object-Oriented C++
    类(构造、析构函数)、封装、继承、多态、虚函数。传引用比传值高效。
    -Template C++
    新的编程范型:模板元编程
  • STL
    容器、迭代器、算法、函数对象。传值更高效。

总结:

C++ 高效编程守则视状况而变化,取决于使用 C++ 的哪一部分。

2. 尽量以const, enum, inline 替换 #define

宁可以编译器替换预处理器。#define不被视为语言的一部分,所以编译出错时难以查找。如:

#define ASPECT_RATIO 1.653
应换为
const double AspectRatio = 1.653

此时AspectRatio会进入编译器的记号表,而且对于浮点数使用常量可能比 #define 码量更小。常量替换 #define 的两种特殊情况:

  1. 定义常量指针

    例如定义一个常量的char *-based字符串:
    改为 string 更合适:
  2. class专属常量

    为了将常量的作用域限制在class中,需将其设为class的一个成员;要确保其只有一份实体,需声明为static成员。
    另外需要提供定义式才能取地址:

    一般形式为:
    但是当这个常量用于指定数组大小,可改用“the enum hack”: 一个属于枚举类型的数值可权充 ints 被使用。
    enum比较像#define, 取enum 地址不合法,但是取const地址合法。enums 与 #defines一样绝不会导致非必要的内存分配。

另一个#define的误用是用它实现宏。如

首先写起来麻烦,而且可能导致不可思议的问题:
改为:

产生一整群函数,每个函数接受两个同型对象。

总结

  • 对于单纯常量,最好以const对象或enums替换#define
  • 对于形如函数的宏,最好改用inline函数替换#define

3. 尽可能使用const

关键词const 用于指定一个不该被改动的对象,加上这个约束后编译器会强制保证实施。所以只要某个值确定应该不变,就应该加上以得到编译器的帮助。

当const和指针组合时:

  • const出现在星号左边——被指物是常量
  • const出现在星号右边——指针自身是常量
  • 两边都有——被指物和指针都是常量

以相对指针的位置来判断,所以以下两种写法相同:


当const和STL迭代器组合时:迭代器的作用就像是 T* 指针。const 迭代器 = T* const 指针,表示这个迭代器不得指向不同的东西,但所指对象的值可变。如:
符号看起来有点易混淆。

当const和函数声明组合时:返回一个常量值

可以防止如下暴行:
显然对于内置类型来说式子不成立,良好的用户自定义类型也应该避免无端的与内置类型不兼容,所以返回const常量可阻止这类错误。

const成员函数

const用于成员函数是为了保证该函数可作用于const对象。这类函数重要的理由:

  1. 使class接口比较容易被理解
  2. 使操作const对象成为可能

两个函数如果只是常量性不同,可以重载。

程序中const对象大多用于传指针或传引用,故以下的调用方式更常见:
调用方式

错误在于企图对由const版的operator[]返回的const char &施行赋值。

const成员函数的两种概念:

  1. bitwise constness认为成员函数只有在不更改对象的任何成员变量时才能称为const。所以const成员函数不能更改对象内任何non-static成员变量。
    但一个更改了指针所指物的成员函数虽然不算const,若只有指针属于对象,那么不会引发编译错误。

最终改变了常量对象的值。

  1. logical constness
    由于存在以上的错误,这里主张一个const成员函数可以修改它所处理的对象内的某些bits。如:
    为了可以修改这两个数据,可用mutable释放non-static成员变量的bitwise constness约束。

在const和non-const成员函数中避免重复

有时候成员函数需要进行多个步骤,这就造成const和non-const成员函数存在大量重复:

解决办法是常量性转除,即用non-const调用const函数。
有两个转型动作,若non-const函数单纯调用operator[]会递归调用自己。所以第一次是 *this转为const,第二次是从const operator[]的返回值中移除const。使用static-cast做安全转型,用const-cast移除const。

值得注意的是反向调用是错误的。用const调用non-const冒着对象被改的风险。

总结

  • 将某些东西声明为const可帮助编译器侦测出错误用法。const可被施加于任何作用域内的对象、函数参数、函数返回类型、成员函数本体。
  • 编译器强制实施bitwise constness,但编写程序时应使用概念上的常量性。
  • 当const和non-const成员函数有着实质等价的实现时,令non-const版本调用const版本可避免代码重复。

4. 确定对象使用前已先被初始化

C++ 变量在声名时有时会被初始化有时不会,需手工完成。

至于内置类型外的东西,要确保每一个构造函数都将对象的每一个成员初始化。但不要混淆赋值与初始化:
而C++规定:对象的成员变量的初始化操作发生在进入构造函数本体之前。较好的写法是用成员初值列:

此时省去了default构造的过程,都进行copy构造。也可在列用default构造:

而且如果成员变量是const或reference就一定需要初值,不能被赋值。所以最简单的做法就是:总是使用成员初值列。

C++ 也有固定的成员初始化次序:基类早于子类,类的成员变量总是以其声明次序被初始化。

non-local static对象的初始化次序

所谓static对象,寿命从被构造出来到程序结束为止。也包括global对象,其中函数内的static对象为local的。它们的析构函数会在main函数结束时自动调用。

现在客户建立如下类处理文件目录:


此时就必须保证tfs在tempDir之前被初始化。正确的做法是 将每个non-local static对象搬到自己的专属函数内,即用local static对象替换non-local static对象。这么做的基础是:C++ 保证函数内static对象会在该函数被调用期间首次遇上该对象的定义式时被初始化。如:

现在使用函数返回的指向static对象的reference,而不再使用static对象自身。
所以为了避免初始化之前使用对象需要做:

  1. 手工初始化内置型non-member对象
  2. 使用成员初值列对付对象的所有成分
  3. 在初始化次序不确定时用函数返回的指向static对象的reference。

总结

  1. 为内置型对象进行手工初始化,C++不保证初始化。
  2. 构造函数最好使用成员初值列,不要再构造函数本体内用赋值操作。初值列列出的变量次序应与声明次序同
  3. 为免除“跨编译单元的初始化次序”问题,用local static对象替换non-local static对象。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,607评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,047评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,496评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,405评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,400评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,479评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,883评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,535评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,743评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,544评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,612评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,309评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,881评论 3 306
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,891评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,136评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,783评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,316评论 2 342

推荐阅读更多精彩内容

  • 1. 让自己习惯C++ 条款01:视C++为一个语言联邦 为了更好的理解C++,我们将C++分解为四个主要次语言:...
    Mr希灵阅读 2,777评论 0 13
  • 前言 把《C++ Primer》[https://book.douban.com/subject/25708312...
    尤汐Yogy阅读 9,500评论 1 51
  • 1 让自己习惯 C++ 条款01:视 C++ 为一个语言联邦 将C++视为一个由相关语言组成的联邦而非单一语言。在...
    暗夜望月阅读 359评论 0 1
  • C++文件 例:从文件income. in中读入收入直到文件结束,并将收入和税金输出到文件tax. out。 检查...
    SeanC52111阅读 2,751评论 0 3
  • 现在开始上班了,很久没有更新了.今天写个collectionView给大家看看,这种样式的collectionvi...
    徐老茂阅读 885评论 3 12