当前位置: 首页 > news >正文

【读书笔记】Effective C++ 条款3:尽可能使用const

目录

1. const的本质:将编译器当作你的朋友去维护一个常量

2. 指针的const

3. 迭代器的const

4. 用const减少调用错误

5. const成员函数

6. bitewise观点

7. 修正:mutable

8. Const和非const写法归一

详细解析

可以const调用非const吗?


1. const的本质:将编译器当作你的朋友去维护一个常量

2. 指针的const

const char *str1 = "hello"; str1 = "ttt"; // 可以改变指针指向

在上面的代码中,const仅仅为保证str1指向的字符不被改变,但是我们可以将str1指向的内容改变。

char str[] = "hello"; char* const str2 = str; str2[0] = 'a'; // 可以修改内容

上面代码中的const只能维持str2指向不变,但是内容可以改变。

因此,我们要将两个const结合:

const char* const str3 = "hello";

尽可能保证常量不被改变。在这过程中,将const作为让编译器维护常量的口令,将编译器当作你的朋友。

但是这一方案也并非天衣无缝(远古版本c++),在下面会提到

3. 迭代器的const

由于迭代器是基于指针的,也就同理,意味着有两种类型的const:const iteratorconst_iterator

  • const iterator相当于char* constT* const),指向不可变

  • const_iterator相当于const char*const T*),内容不可变

4. 用const减少调用错误

class A { public: A(const int a = 1) :_a(a) { } A operator*(const A& aa) { A ret(_a * aa._a); return ret; } A operator=(const A& aa) { _a = aa._a; return *this; } bool operator==(const A& aa) const { return _a == aa._a; } operator bool() { return _a != 0; } private: int _a; }; A a(1); A b(2); A c(3);

在这样一个类中,如果误将if (b == c)写成if (b = c),那么你是不会察觉的。但如果加上const,那么就会报错。

5. const成员函数

在类的函数后加上const有两个作用:

  • 让函数做什么更加容易理解

  • 可以传const对象

6. bitewise观点

首先,假设有一座房子,坏没坏有种标准。bitewise观点就是只要房子外壳还是一样的,但是里面怎么坏都无所谓。这就是编译器对于const的理解:只要这块内存不动,至于内存怎么变无所谓。

这样就会发生神奇的事(远古版本c++):

const char* const str = "hello"; char* s = &str[0]; s[0] = 'a'; cout << str << endl;

(注意:由于书本是比较老的,现在即便是c++98编译也会报错,但是由于是书中的例子,就讲究看一下吧)

同时,对于缓存,我们可能需要微小改动里面的值,即使改变了也依旧认为是const。因此也有bitewise观点认为错误,但我们需要的情况。

7. 修正:mutable

在第六点,我们发现bitewise观点很多时候并不能完全胜任

因此就需要修正

有可能我们在const函数中也需要修改值,加上mutable就可以了。

class A { public: A(int a, int b) :_a(a) ,_b(b) { } int getb() { _b = 100; return _b; } private: int _a; mutable int _b; };

这就是logical constness观点。

8. Const和非const写法归一

比如[]的重载,const函数和非const函数绝大多数代码都是一样的,因此能否复用呢?可以用非const复用const。

namespace bit { class string { public: string(const char* str = "") { _size = strlen(str); _capacity = _size; _str = new char[_capacity + 1]; strcpy(_str, str); } char& operator[](size_t pos) { assert(pos < _size); assert(pos); return _str[pos]; } const char& operator[](size_t pos) const { assert(pos < _size); assert(pos); return _str[pos]; } ~string() { if (_str) { delete[] _str; _str = nullptr; _size = 0; _capacity = 0; } } private: char* _str = nullptr; size_t _capacity = 0; size_t _size = 0; static const size_t npos = -1; }; }

上面是简易的string类。

char& operator[](size_t pos)改为以下代码:

char& operator[](size_t pos) { return const_cast<char&>( static_cast<const bit::string&>(*this) [pos] ); }

详细解析

  1. static_cast<const bit::string&>(*this)
    其中static_cast为c++的安全转换,将*this转为const bit::string的引用。

  2. [pos]调用[]重载,返回const char
    调用const版本的operator[]。

  3. const_cast<char&>
    const_cast将常性去掉,转为char引用。

可以看到,代码繁琐冗长,因此要权衡代码可读性与简洁性。

可以const调用非const吗?

理论可以,但是调用非const就意味着需要冒着被改变的风险。

http://www.jsqmd.com/news/163565/

相关文章:

  • 2025年口碑好的板式平焊不锈钢法兰/带颈平焊不锈钢法兰优质厂商精选推荐(口碑) - 行业平台推荐
  • SMUDebugTool完全解析:AMD系统调校的专家级解决方案
  • 2025年质量好的压力管道不锈钢法兰/盲板不锈钢法兰热门品牌厂家推荐 - 行业平台推荐
  • PyTorch-CUDA-v2.9镜像支持空间站运维决策
  • PyTorch-CUDA-v2.9镜像 + Triton推理服务器:高并发部署方案
  • 如何在Windows和Linux上提升AirPods使用体验?5个实用技巧分享
  • Qwen3-Omni:全能多模态AI交互新体验
  • NVIDIA显卡终极色彩校准指南:让显示器回归真实色彩
  • 金融数据获取实战指南:7个核心技巧快速掌握AKShare工具
  • PyTorch-CUDA-v2.9镜像能否运行Quantization Aware Training量化感知训练?
  • Qwen2.5-VL-3B:小模型大能力,视觉AI新突破!
  • 5分钟搞定:Beyond Compare永久授权终极方案
  • BetterNCM插件管理器完整使用教程:5分钟快速上手指南
  • PyTorch-CUDA-v2.9镜像用于卫星图像解译
  • Altium Designer在工业控制中的应用操作指南
  • 2025年知名的3档钮子开关/3C认证钮子开关品牌厂商推荐(更新) - 行业平台推荐
  • 终极神经网络绘图指南:NN-SVG完整使用教程
  • 智能转换网页内容:高效管理信息的终极快速方案
  • 深度解析FUXA:构建企业级工业物联网平台的完整技术方案
  • PyTorch-CUDA-v2.9镜像中的CUDA版本是多少?驱动要求说明
  • Visual C++运行库终极修复指南:一键解决软件启动失败
  • JPEXS反编译神器进阶指南:从安装到精通的高效工作流
  • 腾讯开源!HunyuanWorld-Voyager:单图打造3D探索视频
  • 基于SpringBoot + Vue的云与糖蛋糕购物平台系统
  • My-TODOs桌面待办工具:5分钟打造高效工作系统的终极指南
  • BetterNCM插件完整安装教程:轻松打造个性化网易云音乐体验
  • GitHub热门项目推荐:基于PyTorch-CUDA-v2.9的开源AI框架
  • 如何快速搭建Android开发环境?这款自动化工具让你3分钟搞定
  • OpenCore配置工具完整使用指南:从安装到高级配置
  • c# 中间表