C++高级进阶 第四季:const详解(二) 常量折叠

简介:

一、文章来由

const详解之二

二、const 代替 #define

const最初动机就是代替 #define。

const 优于 #define:
(1) #define没有类型检查,const在编译期(而不是预编译期)做类型检查;

(2)const方便调试和定位bug。

所以应该完全用const代替#define

三、头文件中的const

(1)要使用const代替#define,同样需要把const定义放进头文件(或其他格式文件,include即可)。这样通过包含头文件,可把const定义单独放在一个地方并把它分配给一个编程单元。

(2)C++中的const默认为内部连接(internal linkage),也就是说 const 仅在被定义过的文件里才可见,而在连接时不能被其他编译单元看到。

但是在c中仅仅在另一个文件中定义(不用extern修饰),另一个文件也是合法的。c中const是必需分配内存的,而c++实际上一开始是不会分配内存的,只是存在字符表中。

//another.c
const int a = 5;

//other.c
#include <stdio.h>

int main()
{
    extern const int a;
    printf("%d\n",a);

    return 0;
}

(3)定义一个const时,必须赋一个值给它,除非用extern做说明:extern const int bufsize;

这里写图片描述

四、常量折叠(constant folding)

通常C++编译器并不为const创建存储空间,相反它把这个定义保存在它的符号表里,但extern强制进行了存储空间分配,取const地址也会**,这也解释了const详解(一)中可以修改const空间,但是cout原值相当于还是去符号表中找。

由于extern意味着使用外部连接(定义时使用extern),因此必须分配存储空间,这也就是说有几个不同的编译单元应当能够引用它,所以必须有存储空间。

通常情况下,当extern不是定义的一部分时,不会分配存储空间。如果使用const,那么编译时会进行常量折叠。

那同文件extern呢

extern const int a;
int main()
{
    //freopen("input.txt","r",stdin);

    cout<<a<<endl;

    return 0;
}

const int a = 66;

因为extern不是定义的一部分,所以个人认为这也不会分配空间。

五、为什么const要内部连接

当然想绝对不为任何const分配存储是不可能的,尤其是复杂的结构。所以const的定义必须默认内部连接,即连接仅在特定的编译单元内;否则,由于众多的const在多个cpp文件分配内存,引起连接错误。

连接程序在多个对象文件看到统一的定义就会“抱怨”。然而,因为const默认内部连接,所以连接程序不会跨过编译单元连接那些定义,因此不会有冲突。


参考资料

[1] c++编程思想

[2] http://blog.csdn.net/bestrivenfan/article/details/50951809

相关文章
|
16天前
|
存储 安全 编译器
第二问:C++中const用法详解
`const` 是 C++ 中用于定义常量的关键字,主要作用是防止值被修改。它可以修饰变量、指针、函数参数、返回值、类成员等,确保数据的不可变性。`const` 的常见用法包括:
59 0
|
5月前
|
C++
【学习笔记】【C/C++】 c++字面值常量
【学习笔记】【C/C++】 c++字面值常量
48 1
|
7月前
|
编译器 C++
C++模板进阶
C++模板进阶
28 1
|
7月前
|
安全 数据库连接 C++
C++一分钟之-C++中的常量与宏定义
【6月更文挑战第18天】**C++编程中,常量(`const`和`constexpr`)确保值不变,用于配置、计算和逻辑控制。宏定义是预处理阶段的文本替换,适用于条件编译和简化代码。注意`const`的正确使用,避免过度的`const_cast`。宏可能导致命名冲突、副作用和类型安全问题,应谨慎使用。优选类型安全的常量定义,以提升代码质量和可维护性。**
89 5
|
7月前
|
Linux C++
c++高级篇(三) ——Linux下IO多路复用之poll模型
c++高级篇(三) ——Linux下IO多路复用之poll模型
|
7月前
|
缓存 监控 网络协议
c++高级篇(二) ——Linux下IO多路复用之select模型
c++高级篇(二) ——Linux下IO多路复用之select模型
|
7月前
|
存储 算法 程序员
【C++进阶】深入STL之 栈与队列:数据结构探索之旅
【C++进阶】深入STL之 栈与队列:数据结构探索之旅
65 4
|
7月前
|
算法 安全 编译器
【C++进阶】模板进阶与仿函数:C++编程中的泛型与函数式编程思想
【C++进阶】模板进阶与仿函数:C++编程中的泛型与函数式编程思想
64 1
|
6月前
|
C++
【C++】string类的使用④(常量成员Member constants)
C++ `std::string` 的 `find_first_of`, `find_last_of`, `find_first_not_of`, `find_last_not_of` 函数分别用于从不同方向查找目标字符或子串。它们都返回匹配位置,未找到则返回 `npos`。`substr` 用于提取子字符串,`compare` 则提供更灵活的字符串比较。`npos` 是一个表示最大值的常量,用于标记未找到匹配的情况。示例代码展示了这些函数的实际应用,如替换元音、分割路径、查找非字母字符等。
|
7月前
|
编译器 C++
【C++】类和对象④(类的默认成员函数:取地址及const取地址重载 )
本文探讨了C++中类的成员函数,特别是取地址及const取地址操作符重载,通常无需重载,但展示了如何自定义以适应特定需求。接着讨论了构造函数的重要性,尤其是使用初始化列表来高效地初始化类的成员,包括对象成员、引用和const成员。初始化列表确保在对象创建时正确赋值,并遵循特定的执行顺序。