C中const 定义常量的使用
- 先看如下代码
#include <stdio.h>
#include <string.h> #define ARRSIZE(a) (sizeof(a)/sizeof(a[0]))
int main()
{
const int i = 10;
int *p = (int *)&i;
int a[i];
*p = 100; printf("%d\n", ARRSIZE(a));
printf("%d\t%d\n", i, *p);
return 0;
}我用gcc编译运行,上面这段简单的代码的输出结果是:100 # 这也是一个很奇怪的结果。100 100
这个结果说明了2个问题,1:可以用const变量作为数组大小;2:可以修改const变量的值。
但难道真是这样吗?又或者说,难道在不同机器,不同编译器下都这样吗?
并不尽然,首先这个程序在VC编译器下,不能通过,VC编译器不允许用const变量作为数组大小。即使把有关数组的给删掉,VC编译后的可执行文件不能执行,发生内存错误。这说明不可以修改const变量。
因此这个是与编译器有关的。
下面要思考的是:为什么gcc编译器能允许这事发生,而VC不允许。对上面的代码略做修改,将const int i = 10放到main函数外面去。#include <stdio.h>
#include <string.h> #define ARRSIZE(a) (sizeof(a)/sizeof(a[0]))
const int i = 10;
int main()
{
int *p = (int *)&i;
int a[i]; printf("%d\n", ARRSIZE(a));
*p = 100;
printf("%d\t%d\n", i, *p);
return 0;
}这时gcc编译通过,运行出现段错误。段错误是经典错误,表明访问了不该访问的段。因此这说明全局的const变量不能修改;局部的const能修改。
作如下推测:gcc下,全局的const变量存储于只读的数据段,因此肯定不能修改。局部的const存储在堆栈段,堆栈段是可读可写的;因此不能避免程序员去修改const。
虽然局部const在堆栈,且堆栈可写,但如下方式却行不通:const int i = 10;i = 100; # 这里的错误是编译错误。因此编译器不允许你对局部const变量[直接]修改,它能够识别你在修改const变量。但正如前面的程序,你只要将局部const变量的地址赋给指针,就能骗过编译器了。而在VC下,const变量总是在只读数据段里。
这时再来看数组#include <stdio.h>
#include <string.h> #define ARRSIZE(a) (sizeof(a)/sizeof(a[0]))
const int i = 10;
int a[i]; // 将数组拿到main的外面来
int main()
{
printf("%d\n", ARRSIZE(a));
return 0;
}这时会有编译错误:error: variably modified ‘a’ at file scope如果将"int a[i]"放到main函数里,则正确。
我还不太清楚这样的具体原因。这里我只能大致得出结论是:不能用const变量来作为数组大小。至于为什么将"int a[i]"放到main函数里时没有错误,我并不清楚,我看了下汇编码,发现gcc对这种情况进行了一些特殊的处理,并不是简单的分配堆栈空间,但更细致的原因,我并未深入研究。
另外在VC下,是不能用const变量来作为数组大小的,不管是局部还是全局的const。
因此不要试着去修改const变量,也不要用const变量来作为数组大小
C中const 定义常量的使用的更多相关文章
- php中const定义常量
const 常量 1.在定义时必须被初始值,2.前面不加任何修饰符3.变量名字母一般都大写4.常量可以被子类继承5.一个常量是属于一个类的,而不是某个对象的 作用:当某些值是固定不变的,就用const ...
- PHP中define()和const定义常量的区别
在PHP中可以通过define()和const两种方式定义常量可是在开发中我们应该什么时候用define()定义常量,什么时候用const定义常量? 这两种方式定义常量的主要区别是什么? 从5.3版本 ...
- 在php中define和const定义常量的区别
define和const都可以用来定义常量,但是const定义常量的时候大小写敏感,而define可以通过设置第三个参数为true的时候来取消大小写敏感! 如图: 引用地址:点这里
- PHP 中 define() 和 const 定义常量时的区别
自 PHP 5.3.0 起,有两种方式定义常量,使用 const 关键字或者 define() 函数: 1 2 const FOO = 'BAR'; define('FOO', 'BAR'); 这 ...
- C++ —— 非类中使用const定义常量的初始化,以及#define和typedef的区别
总结一下在非类中使用const关键字定义常量时的初始化问题,亲测VS2015.顺便记录#define宏和typedef的区别. 1 首先对const声明的常量的初始化做简单小结: , w2 = , w ...
- 为什么很多人使用#define而不是const定义常量
众所周知,C语言一开始只有#define,C程序员用#define定义符号常量.但后来ANSI C加入了const限定符,而const应该比#define更好,为什么现在的C程序员还在大量使用#def ...
- iOS define 宏定义 和 const定义常量区别
const const 是c++中的修饰符. c++中常用来定义常量,修饰左值. #define 宏定义语句, 在预处理阶段直接做文本替换,不做类型检查. 它们之间的最大区别: 1. 对于co ...
- 关于java中接口定义常量和类定义常量的区别
/** * * @author YZJ * @Description java中定义常量的最佳方法 */ public final class Contants{ /** * @Description ...
- js 中const 定义的值是否能更改
const定义的基本类型不能改变,但是定义的对象是可以通过修改对象属性等方法来改变的. 1. const aa=trueaa=falseconsole.log(aa)VM1089:2 Uncaught ...
随机推荐
- Sql 正确删除用户过期的数据
怎样才算是正确的删除过期的数据呢?先交代一下前提,XX网站上面有一个放心企业专区,办理超级会员即可成为放心企业,放心企业可设置推荐职位展示在放心企业专区,信息都是存放在Info表中的,所谓的推荐职位就 ...
- saltStack运维工具的部署及master迁移实现的过程详解
服务器端:192.168.3.87 客户端:192.168.3.86.192.168.3.108 1.salt服务器端安装 192.168.3.87 rpm -Uvh http://mirrors.y ...
- SIP 认证
理解SIP的认证 1. 认证和加密 认证(Authorization)的作用在于表明自己是谁,即向别人证明自己是谁.而相关的概念是MD5,用于认证安全.注意MD5仅仅是个hash函数而已,并不是 ...
- 转载:Java中的字符串常量池详细介绍
引用自:http://blog.csdn.net/langhong8/article/details/50938041 这篇文章主要介绍了Java中的字符串常量池详细介绍,JVM为了减少字符串对象的重 ...
- Ex 6_21 最小点覆盖问题_第八次作业
子问题定义: 对于图中的每个结点,有两种状态,即属于最小点覆盖和不属于最小点覆盖,定义minSet[i][0]表示结点i属于点覆盖,并且以i为根的树的最小点覆盖的大小.minSet[i][1]表示点i ...
- 解析神奇的 Object.defineProperty
这个方法了不起啊..vue.js是通过它实现双向绑定的..而且Object.observe也被草案发起人撤回了..所以defineProperty更有必要了解一下了. 几行代码看他怎么用 var a= ...
- Java基础96 ajax技术的使用
本文知识点(目录): 1.ajax的概念 2.使用ajax技术获取服务端的数据_实例 3.使用ajax技术检查用户名是否已存在_实例 4.使用ajax技术验证登录页面的用户名和密码_实例 ...
- SpringMVC(1):Web MVC简介
原文出处: 张开涛 Web MVC简介 1.1.Web开发中的请求-响应模型: 在Web世界里,具体步骤如下: 1. Web浏览器(如IE)发起请求,如访问 2. Web服务器(如Tomcat)接 ...
- Python 列表推导、迭代器与生成器
1.列表推导 1 2 3 4 5 6 7 8 9 10 11 numbers = [i for i in range(10) if i % 2 == 0] print(numbers) seq = ...
- unittest中更高效的执行测试用例一个类只需要打开一次浏览器
示例代码 baidu.py # _*_ coding:utf-8 _*_ import csv,unittest #导入csv模块 from time import sleep from seleni ...