C++ 核心指南之 C++ P.哲学/基本理念(上)
C++ 核心指南(C++ Core Guidelines)是由 Bjarne Stroustrup、Herb Sutter 等顶尖 C+ 专家创建的一份 C++ 指南、规则及最佳实践。旨在帮助大家正确、高效地使用“现代 C++”。
这份指南侧重于接口、资源管理、内存管理、并发等 High-level 主题。遵循这些规则可以最大程度地保证静态类型安全,避免资源泄露及常见的错误,使得程序运行得更快、更好。
文中提到的 GSL(Guidelines Support Library) 是 C++ 核心指南支持库 https://github.com/Microsoft/GSL
P:Philosophy 基本理念
本节的规则反映了现代 C++ 的哲学/基本理念,贯穿整个 C++ 核心指南:
规则摘要:
- P.1:直接用代码表达想法
- P.2:编写符合 ISO 标准的 C++ 代码
- P.3:表达意图
- P.4:(理想情况下)程序应该是静态类型安全的
- P.5:优先使用编译时检查而不是运行时检查
- P.6:无法在编译时检查的内容应该在运行时可检查
- P.7:尽早捕获运行时错误
- P.8:不要泄露任何资源
- P.9:不要浪费时间或空间
- P.10:优先使用不可变数据而不是可变数据
- P.11:封装混乱的结构,而不是让其散布在代码中
- P.12:根据需要使用支持工具
- P.13:根据需要使用支持库
这些基本理念是其他章节具体规则的理论基础。
P.1:直接用代码表达想法
关联 P.3
- 代码不会说谎,注释可不一定
- 编译器不看注释(很多程序员也不看 )
- 代码有确定性的语意,编译器和工具可以帮助检查
例子
class Date {
public:
Month month() const; // 好
int month(); // 不好
};
第一个 month() 声明很明确地表达了返回 Month 对象,并且不会修改 Data 对象的状态
第二个 month() 只能让读者去猜,也更容易产生 bug
例子
直接使用语言特性,代码冗长,且不能直接表达意图
void f(vector<string>& v)
{
string val;
cin >> val;
int index = -1; // 而且应该用 gsl::index
for (int i = 0; i < v.size(); ++i) {
if (v[i] == val) {
index = i;
break;
}
}
}
实现同样功能,用标准库则可以更清晰地表达意图
void f(vector<string>& v)
{
string val;
cin >> val;
auto p = find(begin(v), end(v), val); // better
}
使用(设计良好的)库比直接使用语言特性能更好地表达意图(要做什么,而不是怎么做)。
开发者应该熟悉 C++ 标准库以及项目的基础库。如果使用 C++ 核心指南,还应该知道 GSL(Guidelines Support Library),并且在适当的时候使用。
例子
// :s 是代表什么?
change_speed(double s);
change_speed(2.3);
更好的做法是明确 double 的含义(绝对速度/变化值)以及使用的单位:
change_speed(Speed s); // 稍好一些,s 代表速度绝对值
change_speed(2.3); // 错误:没有单位
change_speed(23_m / 10s); // 单位 m/s
也可以用不带单位的 double 作为速度变化值,但这样容易出错。更好的做法是定义 Delta 类型。
代码检查建议
如果不修改对象/参数,使用 const 明确表达意图
- 成员函数是否修改对象状态
- 函数是否修改传入的指针或引用参数
警惕强制类型转换,强制类型转换使得编译器无法帮你进行类型检查。一般来说,强制类型转换意味着程序设计的缺陷
检查和标准库行为类似的代码:例如很多循环都可以用标准库算法替代
P.2:编写符合 ISO 标准的 C++ 代码
这份指南是针对 ISO 标准 C++ 的。
有的环境下需要扩展,如访问系统资源。这种情况下应该限制扩展的使用范围。如果可以,用接口封装扩展,以便在不支持扩展特性的系统上关闭或屏蔽这部分代码
(标准中没有规定的)扩展通常没有严格的定义,即使是那些常见的、被大多数编译器支持的扩展特性(如 #pragma once),在某些边界条件下,行为可能会有微小差异。
扩展会影响代码可移植性;但完全遵守 ISO 标准并不能保证可移植性
避免未定义的行为,如表达式的求值顺序
int i = foo() * bar();其中 foo() 和 bar() 的调用顺序未定义!- 参考 undefined order of evaluation
了解“取决于实现”的含义,如 C++ 标准只规定了 int 类型的最小大小,并没有规定确切的大小,不同的 C++ 实现可以采用不同的大小
有的环境需要限制使用某些 C++ 标准库和特性,如汽车、航空领域中的禁用动态内存分配、异常机制等
代码检查建议
使用最新的 C++ 编译器,在编译选项中关闭扩展特性
P.3:表达意图
关联 P.1
除非通过命名或者注释表述了某些代码的意图,否则很难判断代码是否在做正确的事
例子
gsl::index i = 0;
while (i < v.size()) {
// ... do something with v[i] ...
}
上述代码中,遍历 v 中元素的意图没有很好地表达出来。下标索引 i 暴露出来,且生命周期超出循环体,可能被误用。更好的做法:
for (const auto& x : v) { /* do something with the value of x */ }
改进后的代码不涉及具体迭代机制(如下标索引),并且循环操作的是一个 const 引用,不会意外地修改 v 中数据。如果的确需要修改,可以改为非 const 引用:
for (auto& x : v) { /* 修改 x */ }
关于 for 语句详见条款 ES.71。更好的做法是用使用标准库的算法,例如 for_each,可以直接地表达意图:
for_each(v, [](int x) { /* do something with the value of x */ });
for_each(par, v, [](int x) { /* do something with the value of x */ });
这个版本还传达了我们不关心处理 v 中元素的顺序的意图。
程序员应该熟悉:
注
- 说明要做什么,而不是怎么做
- 有些编程语言在表达意图方面比其他语言做得更好
例子
如果用 2 个 int 表示一个点的二维坐标:
// (x1,y1,x2,y2) 还是 (x,y,h,w)?
// 不明确,需要查文档
draw_line(int, int, int, int);
// 更清晰
draw_line(Point, Point);
代码检查建议
有些常见的模式有更好的替代方案:
- 简单的 for 循环 ️ 范围 for 循环
f(T*, int)接口 ️f(span<T>)- 作用域很大的循环控制变量
- 裸的 new 和 delete
- 参数过多的函数
有工具可以(半)自动检测并给出修改建议(如 clangd)
P.4:(理想情况下)程序应该是静态类型安全的
理想强况下,编译器在编译过程中可以进行静态类型检查,程序应该是(编译期)类型安全的。实际上是不可能的,因为:
- unions
- 类型转换
- 数组退化
- 范围错误
- 窄化转换
注
上述几点也是严重问题的根源(如程序崩溃、安全漏洞)。C++ 在尝试用一些替代技术来规避上述问题
代码检查建议
- union ️ C++ 17 variant
- cast ️ 尽量避免使用,模版可以解决一部分问题
- 数组退化 ️ 使用 span(GSL)
- 范围错误 ️ 使用 span
- 窄化转换 ️ 尽量避免使用,如果必要可以使用 GSL 中的 narrow 或者 narrow_cast
P.5:优先使用编译期检查而不是运行期检查
不需要针对运行期错误编写错误处理代码,代码更清晰,性能也更好。
例子
// Int 是整数类型别名
// 运行期检查,不要这么做
int bits = 0;
for (Int i = 1; i; i <<= 1)
++bits;
if (bits < 32)
cerr << "Int too small\n";
这段代码并没有达到它的目的,因为 C++ 标准没有规定溢出的行为。即溢出的行为是未定义的,不同平台、不同编译器的行为可能并不相同!
应该用更简单的 static_assert 来替代:
// Int 是整数类型别名
// OK:编译期检查
static_assert(sizeof(Int) >= 4);
更好的做法是直接用 int32_t
例子
// 读取最多 n 个整数到 *p
void read(int* p, int n);
int a[100];
read(a, 1000); // 越界
更好的做法:
// 读到一个 span 中
void read(span<int> r);
int a[100];
read(a); // better: 让编译器自动计算元素数量
总之,不要把编译期能做的事推迟到运行时!
代码检查建议
- 检查参数中是否有指针
- 检查代码中是否有运行时范围检查
P.6:无法在编译时检查的内容应该在运行时可检查
理想情况下,应该捕获所有错误,要么在编译期,要么在运行时。但实际上不可能在编译期捕获所有的错误,而想在运行期捕获其余所有的错误的代价也很大,但是我们还是要尽量去检查可能的错误,否则可能导致错误的结果或者程序崩溃。
反面例子
// 独立编译,可能动态加载
extern void f(int* p);
void g(int n)
{
// 元素数量没传给 f
f(new int[n]);
}
这里元素的数量这一关键信息没有传递给 f,并且这种情况静态代码分析工具可能无法发现,即使动态检查可能也会非常困难,特别是当 f() 是 ABI 的一部分时。这么设计代码只会让错误检测更困难。
反面例子
当然,可以把元素数量和指针一起传递
// 独立编译,可能动态加载
extern void f2(int* p, int n);
void g2(int n)
{
// 还是会有传错 m 的可能
f2(new int[n], m);
}
一种很常见的做法是把元素数量作为单独的参数传递,这样要比只传指针、然后通过其他(没有明确说明的)途径获取元素数量要好一些。但即便如此,一个小的手误(如把 n 按成 m)就可能导致严重的问题。f2() 的两个参数之间的联系只是一种约定,但并不够明确。
此外,f2() 应该 delete[] 参数这一点也没有明确说明(或者本该由 f2() 的调用者负责 delete[] 但是忘记了?)
反面例子
标准库的智能指针并不能传递元素的数量:
// 独立编译,可能动态加载
// 假设调用代码和标准库是 ABI 兼容的,用兼容的 C++编译器编译
extern void f3(unique_ptr<int[]>, int n);
void g3(int n)
{
// 分开传递指针和大小
f3(make_unique<int[]>(n), m);
}
正面例子
把指针和元素的数量作为一个整体传递:
// 独立编译,可能动态加载
// 假设调用代码和标准库是 ABI 兼容的,用兼容的 C++编译器编译
extern void f4(vector<int>&);
extern void f4(span<int>);
void g3(int n)
{
vector<int> v(n);
f4(v); // 传引用,保留所有权
f4(span<int>{v}); // 传视图,保留所有权
}
元素数量是对象的一部分,不容易出错,且总是可以在运行时检查。
例子
如何同时传递所有权和校验所需的所有信息?
// OK: 移动 vector
vector<int> f5(int n)
{
vector<int> v(n);
// ... initialize v ...
return v;
}
// 丢失大小信息 n
unique_ptr<int[]> f6(int n)
{
auto p = make_unique<int[]>(n);
// ... initialize *p ...
return p;
}
// 丢失大小信息 n,并且用完可能忘记删除
owner<int*> f7(int n)
{
owner<int*> p = new int[n];
// ... initialize *p ...
return p;
}
例子
- TODO:需要增加一个例子
- show how possible checks are avoided by interfaces that pass polymorphic base classes around, when they actually know what they need? Or strings as “free-style” options
代码检查建议
- 标记 (指针,数量) 这种接口(可能会有很多因为兼容性原因无法修复)
- ...
C++ 核心指南之 C++ P.哲学/基本理念(上)的更多相关文章
- 三年磨一剑,robot framework 自动化测试框架核心指南,真正讲透robot framework自动化测试框架(笔者新书上架)。
序 关于自动化测试的工具和框架其实有很多.自动化测试在测试IT行业中扮演着越来越重要的角色,不管是在传统的IT行业还是高速发展的互联网行业或是如今的大数据和大热的人工智能领域,都离不开测试,也更加离不 ...
- Robot Framework自动化测试框架核心指南-如何使用Java编写自定义的RobotFramework Lib
如何使用Java编写自定义的RobotFramework Lib 本文包括2个章节 1. Robot Frdamwork中如何调用java Lib库 2.使用 java编写自定义的Lib 本文作者为: ...
- Robot Framework自动化测试框架核心指南-如何做好自动化测试平台框架的设计
自动化测试如果需要能高效快速的支撑软件项目的测试,项目的快速迭代以及上线,除了以上我们介绍的需要许多的Lib来支持以及需要高效的去编写自动化测试案例外,还需要一个好的自动化测试框架平台来支撑我们的自动 ...
- [译] JavaScript核心指南(JavaScript Core) 【转】
本文转自:http://remember2015.info/blog/?p=141#scope-chain 零.索引 对象(An Object) 原型链(A Prototype Chain) 构造函数 ...
- Java多线程编程模式实战指南一:Active Object模式(上)
Active Object模式简介 Active Object模式是一种异步编程模式.它通过对方法的调用与方法的执行进行解耦来提高并发性.若以任务的概念来说,Active Object模式的核心则是它 ...
- Kubernetes 降本增效标准指南 | 基于K8s 扩展机制构建云上成本控制系统
作者 王玉君,腾讯云后台高级开发工程师,负责腾讯云原生系统开发及建设. 晏子怡,腾讯云容器产品经理,在K8s弹性伸缩.资源管理领域有丰富的实战经验. 导语 Kubernetes 作为 IaaS 和 P ...
- ASP.NET Core 中文文档 第二章 指南(5) 在 Nano Server 上运行ASP.NET Core
原文 ASP.NET Core on Nano Server 作者 Sourabh Shirhatti 翻译 娄宇(Lyrics) 校对 刘怡(AlexLEWIS).许登洋(Seay).谢炀(kile ...
- GNU 项目(开源社区的由来,背后的哲学)
转自译言网:http://article.yeeyan.org/view/88497/59257/ 第一个软件共享社区 当我在1971年开始在麻省理工人工智能实验室工作时, 我成为一个已经存在多年的软 ...
- 编程哲学之C#篇:01——创世纪
我们能否像神一样地创建一个世界? 对于创建世界而言,程序员的创作能力最接近于神--相对于导演,作家,漫画家而言,他们创建的世界(作品)一旦完成,就再也不会变化,创建的角色再也不会成长.而程序员创建的世 ...
- Spring boot参考指南
介绍 转载自:https://www.gitbook.com/book/qbgbook/spring-boot-reference-guide-zh/details 带目录浏览地址:http://ww ...
随机推荐
- 深度学习03-(图像梯度处理、图像轮廓、图像预处理在AI中的应用)
深度学习03-计算机视觉基本理论2 深度学习03-(计算机视觉基本理论2) 图像梯度处理 什么是图像梯度 模板运算 均值滤波 高斯滤波 中值滤波 边沿检测 锐化 图像轮廓 什么是图像轮廓 查找和绘制轮 ...
- 希望所有计算机学生能看到这篇c语言教程
大部分程序员走入编程世界第一个学习的语言就是C语言. 作为一门古老的编程语言,c语言拥有48年的发展历程. 为什么要学习 C语言? C语言是学习计算机程序设计语言的入门语言.最全面的编程面试网站 C语 ...
- 使用require.context实现优雅的预加载
前言 在前端开发中,对页面花里胡哨度[注1]要求越高的页面,用到的图片.音频什么的就越多,比如什么结婚请柬.展会请柬.发布会宣传页.数据大屏.虽然现在浏览器不允许网页在没有用户交互的情况下播放音频,但 ...
- Prism Sample 10 10-CustomRegistrations
作用同上节,这里是用修改注册的方式自定义View和ViewModel的关联. protected override void ConfigureViewModelLocator() { base.Co ...
- elSelect点击空白处无法收起下拉框(失去焦点并隐藏)
学习记录,为了以后有同样的问题,省得再百度了,方便自己也方便你们element 中多选的select 有个问题,就是点击空白或者关闭弹窗,下拉还会一直展示出来百度了好一会,觉得下面两位大佬说的最合理, ...
- 文心一言 VS chatgpt (2)-- 算法导论1.2
一.给出在应用层需要算法内容的应用的一个例子,并讨论涉及的算法的功能. 文心一言: 在应用层需要算法内容的应用的一个例子是无人机自主飞行控制.无人机自主飞行控制需要算法来确定无人机的位置.速度和方向, ...
- 2022-05-22:给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。 百度百科中最近公共祖先的定义为:“对于有根树 T 的两个节点 p、q,最近公共祖先表示为一个节点 x,满足 x 是 p
2022-05-22:给定一个二叉树, 找到该树中两个指定节点的最近公共祖先. 百度百科中最近公共祖先的定义为:"对于有根树 T 的两个节点 p.q,最近公共祖先表示为一个节点 x,满足 x ...
- 2022-02-01:粉刷房子 II。 假如有一排房子,共 n 个,每个房子可以被粉刷成 k 种颜色中的一种,你需要粉刷所有的房子并且使其相邻的两个房子颜色不能相同。 当然,因为市场上不同颜色油漆的价
2022-02-01:粉刷房子 II. 假如有一排房子,共 n 个,每个房子可以被粉刷成 k 种颜色中的一种,你需要粉刷所有的房子并且使其相邻的两个房子颜色不能相同. 当然,因为市场上不同颜色油漆的价 ...
- 2021-10-28:打家劫舍 II。你是一个专业的小偷,计划偷窃沿街的房屋,每间房内都藏有一定的现金。这个地方所有的房屋都 围成一圈 ,这意味着第一个房屋和最后一个房屋是紧挨着的。同时,相邻的房屋装
2021-10-28:打家劫舍 II.你是一个专业的小偷,计划偷窃沿街的房屋,每间房内都藏有一定的现金.这个地方所有的房屋都 围成一圈 ,这意味着第一个房屋和最后一个房屋是紧挨着的.同时,相邻的房屋装 ...
- 2021-10-14:被围绕的区域。给你一个 m x n 的矩阵 board ,由若干字符 ‘X‘ 和 ‘O‘ ,找到所有被 ‘X‘ 围绕的区域,并将这些区域里所有的 ‘O‘ 用 ‘X‘ 填充。力扣1
2021-10-14:被围绕的区域.给你一个 m x n 的矩阵 board ,由若干字符 'X' 和 'O' ,找到所有被 'X' 围绕的区域,并将这些区域里所有的 'O' 用 'X' 填充.力扣1 ...