对常变量来说c和c++的编译政策不一样

我们在学习中会遇到下面这种问题:

当我们的程序用c编译器编译时下面的代码是不能通过的,即在c语言中学习的变量名不能在数组后缀中使用

int main()
{
    const int n = 10;
    int arr[n] = { 1,2,3 }; //c不能编译通过,在c里面常变量不能作为数组后缀使用
}

而这种编译在c++中可以编译通过

今天,总结一下关于c和c++编译器对常变量不同的编译规则,下面看这样一段代码(c++编译):

int main()
{
    const int a = 10; //只可读不可写
    int b = 0;
    int* ip = (int*)&a;
    *ip = 100;
    b = a; 
    printf("a = %d b = %d *ip = %d", a, b, *ip);
    return 0;

通过运行这段代码,我们得到了下边的运行结果:  

运行结果:a = 10 b = 10 *ip = 100 

在代码中,我们用了耍流氓的方式将a这个常变量的值改为了100,之后将a的值赋给b,按正常情况来说,a和b的值打印出来应该是100,那为什么只有*ip的值是100呢?

下面看一段反汇编:

    int* ip = (int*)&a;
00007FF7E55B195B  lea         rax,[a]  
00007FF7E55B195F  mov         qword ptr [ip],rax  
    *ip = 100;
00007FF7E55B1963  mov         rax,qword ptr [ip]  
00007FF7E55B1967  mov         dword ptr [rax],64h  
    b = a;
00007FF7E55B196D  mov         dword ptr [b],0Ah   

在这段反汇编中可以发现,在我们试图将a的值赋给b时,实际上是(dword ptr [b],0Ah),即直接将10赋给b,这是什么原因呢?

在我们整个程序运行阶段,要经过预编译、编译、汇编、链接这几个过程,实际上c++编译器在预编译过程中,将所有的常变量已经替换成常量表示,在这个程序中就是把a用10代替,这就是为什么会有上边的运行结果,同时也是为什么用常变量做数组后缀可以通过的原因。

当我们把上述代码用c编译器运行时,可以得到以下结果:

运行结果 :a = 100 b = 100 *ip = 100 

在这次运行中,所有的结果都是100,从而证实了c编译和c++编译的不同之处。

总结:

1、实际上,在c语言中对于数组后缀不能使用变量来定义这种说法是不准确的,应该是在C语言中,数组的大小必须是编译时常量(即,在编译时其值必须是已知的),而不能是运行时才能确定的变量。

2、对于常变量c编译是对地址进行操作,对于c++来说是将所有用到常变量的地方换成值(除了取地址)

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值