1. 先看如下代码
  2. #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 定义常量的使用的更多相关文章

  1. php中const定义常量

    const 常量 1.在定义时必须被初始值,2.前面不加任何修饰符3.变量名字母一般都大写4.常量可以被子类继承5.一个常量是属于一个类的,而不是某个对象的 作用:当某些值是固定不变的,就用const ...

  2. PHP中define()和const定义常量的区别

    在PHP中可以通过define()和const两种方式定义常量可是在开发中我们应该什么时候用define()定义常量,什么时候用const定义常量? 这两种方式定义常量的主要区别是什么? 从5.3版本 ...

  3. 在php中define和const定义常量的区别

    define和const都可以用来定义常量,但是const定义常量的时候大小写敏感,而define可以通过设置第三个参数为true的时候来取消大小写敏感! 如图: 引用地址:点这里

  4. PHP 中 define() 和 const 定义常量时的区别

    自 PHP 5.3.0 起,有两种方式定义常量,使用 const 关键字或者 define() 函数:   1 2 const FOO = 'BAR'; define('FOO', 'BAR'); 这 ...

  5. C++ —— 非类中使用const定义常量的初始化,以及#define和typedef的区别

    总结一下在非类中使用const关键字定义常量时的初始化问题,亲测VS2015.顺便记录#define宏和typedef的区别. 1 首先对const声明的常量的初始化做简单小结: , w2 = , w ...

  6. 为什么很多人使用#define而不是const定义常量

    众所周知,C语言一开始只有#define,C程序员用#define定义符号常量.但后来ANSI C加入了const限定符,而const应该比#define更好,为什么现在的C程序员还在大量使用#def ...

  7. iOS define 宏定义 和 const定义常量区别

    const   const 是c++中的修饰符.  c++中常用来定义常量,修饰左值. #define 宏定义语句, 在预处理阶段直接做文本替换,不做类型检查. 它们之间的最大区别: 1.  对于co ...

  8. 关于java中接口定义常量和类定义常量的区别

    /** * * @author YZJ * @Description java中定义常量的最佳方法 */ public final class Contants{ /** * @Description ...

  9. js 中const 定义的值是否能更改

    const定义的基本类型不能改变,但是定义的对象是可以通过修改对象属性等方法来改变的. 1. const aa=trueaa=falseconsole.log(aa)VM1089:2 Uncaught ...

随机推荐

  1. Linux mmc framework2:基本组件之queue

    1.前言 本文主要介绍card下queue组件的主要流程,在介绍的过程中,将详细说明和queue相关的流程,涉及到其它组件的详细流程再在相关文章中说明. 2.主要数据结构和API 2.1 struct ...

  2. centos环境无法安装paramiko的问题解决

    yum install openssl-devel yum install pycrypto yum install python-devel 全部安装完毕后执行pip install paramik ...

  3. 转载: 使用vue全家桶制作博客网站 HTML5 移动网站制作的好教程

    使用vue全家桶制作博客网站   前面的话 笔者在做一个完整的博客上线项目,包括前台.后台.后端接口和服务器配置.本文将详细介绍使用vue全家桶制作的博客网站 概述 该项目是基于vue全家桶(vue. ...

  4. Javascript之BOM与DOM讲解

    一.Javascript组成 JavaScript的实现包括以下3个部分: ECMAScript(核心) 描述了JS的语法和基本对象. 文档对象模型 (DOM) 处理网页内容的方法和接口 浏览器对象模 ...

  5. rem布局加载闪烁问题

    说明:以下内容来自CSDN,如有侵权,请立刻联系博主(我),我将删除该内容. 原文链接  https://blog.csdn.net/u013778905/article/details/779387 ...

  6. 为什么js中0.1+0.2不等于0.3,怎样处理使之相等?(转载)

    为什么js中0.1+0.2不等于0.3,怎样处理使之相等? console.log(0.1+0.2===0.3)// true or false?? 在正常的数学逻辑思维中,0.1+0.2=0.3这个 ...

  7. java 持有对象 ListIterator用法

    package ch07; import java.io.*; import java.util.Iterator; import java.util.LinkedList; import java. ...

  8. php工厂模式的实例

    * 单例模式:用于创建单一类型的唯一实例对象 * 工厂模式:用于创建多种类型的多个实例对象 //声明形状类 class Shape { //声明静态方法create,根据容器形状不同,创建不同图形类的 ...

  9. 区间dp好题cf149d 括号匹配

    见题解链接https://blog.csdn.net/sdjzping/article/details/19160013 #include<bits/stdc++.h> using nam ...

  10. 使用spring-boot-starter-data-jpa 怎么配置使运行时输出SQL语句

    在 application.properties 中加入以下配置 spring.jpa.show-sql=true