常量和常量表达式1
一、基础定义C/C通用核心定义1. 常量Constant程序整个生命周期内值不可修改、固定不变的量是值的实体单个固定值/命名固定值其值的确定时机可在编译期/预处理期C语言中部分const常量值确定于运行期。核心特征只读不可修改强行赋值会触发编译报错是构成常量表达式的基础操作数分为字面常量、命名常量两类是“静态的固定值”。2. 常量表达式Constant Expression编译阶段就能被编译器计算出唯一、固定常量值的表达式由常量和编译器支持的编译期运算符组成无任何运行时计算行为其结果必然是一个常量。核心要求必须同时满足所有操作数都是常量字面常量、枚举常量、C的const常量等仅使用编译期可执行运算符/-/*//||/等排除函数调用、运行时指针运算、变量引用计算结果唯一且固定无任何运行时不确定性。3.常量和常量表达式的对比对比维度常量常量表达式本质固定值的实体编译期求值的计算式形态单个值/命名量10、a、PI常量运算符的组合105、PI*2求值时机编译/预处理/运行期C仅编译期强制求值结果属性自身就是固定值计算结果为常量语法要求无强制专属场景可直接用部分语法位置强制要求内存依赖部分常量需分配内存C的const无内存依赖编译期直接求值4.常量分类字面量、宏常量、枚举常量、const)常量种类示例值确定阶段是否占内存C 中是否是常量表达式C 中是否是常量表达式整型字面量5、0x10编译期不占✅ 是✅ 是字符字面量A编译期不占✅ 是✅ 是字符串字面量hello编译期rodata 区地址运行期非整数常量表达式同左宏常量#define N 5预处理期不占✅ 是✅ 是枚举常量enum { A5 };编译期不占✅ 是✅ 是C 的 const 常量const int a5;运行期栈/全局内存❌不是—C 的 const 常量const int a5;编译期默认不占—✅ 是5. 常量表达式的例子必须满足编译期就能算出唯一值不依赖任何运行时信息。表达式示例在 C 中在 C 中原因5 3✅ 是✅ 是纯字面量运算N * 2宏✅ 是✅ 是预处理替换为字面量A 1枚举✅ 是✅ 是编译期已知a 2C const❌ 不是✅ 是Ca 是运行期只读变量a 2C const—✅ 是Ca 是编译期常量getVal()❌ 不是❌ 不是函数调用运行时行为x 1❌ 不是❌ 不是地址运行期才确定字面量 / 宏 / 枚举都是编译期/预处理期定值→ 可以构成常量表达式。二、区分和判断【常量的表达式】和【常量表达式】1. 常量的表达式只是口语表达式里用了常量比如 C 语言中a10a 是 const它是“常量的表达式”但不是语法意义上的常量表达式。2. 常量表达式是严格语法概念编译期就能算出值不需要运行、不需要读内存。一句话常量表达式一定是常量的表达式但常量的表达式不一定是常量表达式。3、判断预处理/编译期就知道值 → 能进常量表达式要运行、要分配内存、要读地址 → 不能进常量表达式C 的 const ≈ 运行期只读变量C 的 const ≈ 编译期真正常量三、C vs C常量 常量表达式的核心差异重点C和C对const修饰的常量、常量表达式的语义解析、编译期处理、内存分配存在本质差异这是跨语言开发的核心坑点核心差异源于C对const进行了语义升级。1. 核心差异以const int a 5;为例对比维度C语言C语言默认不取aconst int a5的本质运行期只读变量命名常量编译期真正常量常量表达式编译期是否知道a的值❌ 不知道仅做语法校验✅ 知道直接将a与5绑定存入符号表内存分配时机✅ 运行期栈上分配4字节内存❌ 不分配内存无地址取a时才分配使用a的方式运行期从栈内存读取值5编译期直接替换为5零运行开销a是否为常量表达式❌ 不是✅ 是能否用于数组长度定义❌ 非法C90C99仅支持VLA✅ 合法能否用于case标签❌ 非法✅ 合法2. 常量的分类处理差异1字面常量/枚举常量/宏常量C/C无差异字面常量10、3.14、‘a’、枚举常量enum {RED1};、宏常量#define PI 3.14C/C均视为编译期常量可作为常量表达式的操作数宏常量C/C均在预处理期文本替换编译期求值无类型无内存分配。2const修饰的命名常量C/C核心差异C语言const只读限制符仅限制变量不可修改不改变“变量”本质运行期分配内存值在运行期写入因此不是常量表达式C语言const常量定义符若用常量表达式初始化如字面量5则视为编译期常量默认不分配内存编译期值替换因此是常量表达式若取地址a则会在运行期分配内存兼容C语言的内存模型。3. 常量表达式的解析差异操作数支持C语言的常量表达式仅支持字面常量、枚举常量、宏常量C的常量表达式还支持const常量编译期初始化、constexpr常量语法场景支持C语言的常量表达式使用场景极少仅枚举、静态变量初始化C拓展了大量场景数组长度、case标签、模板参数、constexpr/consteval函数C11新增关键字C11引入constexpr显式定义常量表达式、C20引入consteval强制编译期求值C语言无对应关键字始终仅支持基础常量表达式。四、C/C中常见问题 解析问题1C语言中const int a5; int arr[a];为什么报错原因C语言中a是运行期只读变量不是常量表达式而C90标准要求数组长度必须是编译期常量表达式C99支持变量长度数组VLA但这是编译器扩展非标准常量表达式用法且移植性差。解决C语言中用宏常量/枚举常量定义数组长度#define N 5; int arr[N];或enum {N5}; int arr[N];。问题2C中const int a getval(); int arr[a];为什么报错原因C中const常量成为常量表达式的前提是用编译期常量表达式初始化而getval()是函数调用运行期才求值因此a是运行期只读变量不是常量表达式无法用于数组长度。结论C的const常量并非一定是常量表达式初始化值的类型是关键。问题3C/C中#define N 102; int aN*5;结果为什么是20不是60原因宏常量是纯文本替换不是常量表达式的“编译期求值”替换后为int a102*5;按运算符优先级计算为20。解决宏常量定义时加括号#define N (102)替换后为(102)*5编译期求值为60。问题4C中取const int a5;的地址a后a还是常量表达式吗结论使用上仍视为常量表达式内存分配仅为“编译器兼容行为”。解析C中取a时编译器会在运行期为a分配栈内存但普通使用a时仍会编译期值替换仅取地址/指针操作时才读内存不影响a作为常量表达式的语法使用如int arr[a];仍合法。问题5C11的constexpr和普通const有什么区别const隐式常量表达式仅当用编译期常量初始化时才是常量表达式否则为运行期只读变量constexpr显式常量表达式强制编译期求值初始化值必须是常量表达式且自身可作为更复杂常量表达式的操作数支持函数/类的常量表达式定义用法追求明确性时用constexpr如constexpr int a5;比const int a5;更直观编译器校验更严格。问题6常量表达式中可以包含函数调用吗C语言❌ 绝对不能函数调用是运行期行为编译期无法求值C11✅ 可以但必须是constexpr函数编译期可求值的函数如constexpr int add(int x,int y){return xy;} constexpr int aadd(1,2);。五、总结极简记忆1. 通用结论常量是固定值的实体常量表达式是编译期求值的计算式常量是常量表达式的基础常量表达式的结果是常量字面常量、枚举常量、宏常量是C/C通用的编译期常量可构成常量表达式。2. C语言结论const只读变量有内存、运行期定值不是常量表达式常量表达式仅支持字面/枚举/宏常量使用场景极少数组长度、case标签不支持任何const常量仅支持宏/枚举常量。3. C语言结论const编译期初始化编译期常量无内存、编译期定值是常量表达式拓展了常量表达式的使用场景支持数组长度、case标签、模板参数C11用constexpr显式定义常量表达式功能更强大、语法更明确。4. 跨语言开发关键写兼容C/C的代码时避免用const常量定义数组长度、case标签统一用宏常量/枚举常量C独开发时可放心用const/constexpr常量作为常量表达式提升代码可读性和类型安全性宏常量无类型。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2461024.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!