我们在学习中会遇到下面这种问题:
当我们的程序用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++来说是将所有用到常变量的地方换成值(除了取地址)