这到底是为什么? 为什么? 为什么?
用 int* pa 指向 一个char a = 'a';
*pa = 1234567890;
*pa占4个字节, 而char只占1个字节,
请问, 在栈里面, 变量a 后面的内存会被*pa 占用吗?
实事证明, 没有.
那么, *pa的值到底存在哪儿?
#include <iostream>using namespace std;int main(int argc, const char * argv[]){ //a的地址: ebp-0x01, 占一个字节 char a = 'a'; //b的地址: ebp-0x02, 占一个字节 char b = 'b'; //指向a int* pa = (int*)&a; //2^31 - 1, *pa占4字节 *pa = 2147483647; cout<<"a= "<<a<<"\n"; //b没有被覆盖, 仍输出'b' cout<<"b= "<<b<<"\n"; //但是, *pa 4个字节的值却被完整保留, (而不是被char那一个字节截取) //*pa的值到底存在哪儿? cout<<"*pa="<<*pa<<endl; return 0;}#include <iostream>#include<stdio.h>using namespace std;char format[]="%d\n";char ebpformat[]="%x\n";int main(){ char a = 'a'; char b='b'; int* pa = (int*)&a; *pa = 2147483647; cout<<"*pa="<<*pa<<endl; printf("%x %x %x %x %x\n",a,&a,&b,pa,&pa); __asm{ lea eax,dword ptr [ebp] push eax lea eax,ebpformat push eax call printf mov eax,dword ptr [ebp]-4 push eax lea eax,format push eax call printf pop eax pop eax pop eax pop eax } return 0;}
[解决办法]
VC调试时按Alt+8、Alt+6和Alt+5,打开汇编窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应内存和寄存器变化,这样过一遍不就啥都明白了吗。
对VC来说,所谓‘调试时’就是编译连接通过以后,按F10或F11键单步执行一步以后的时候,或者在某行按F9设了断点后按F5执行停在该断点处的时候。
(Turbo C或Borland C用Turbo Debugger调试,Linux或Unix下用GDB调试时,看每句C对应的汇编并单步执行观察相应内存和寄存器变化。)
想要从本质上理解C指针,必须学习汇编以及C和汇编的对应关系。
从汇编的角度理解和学习C语言的指针,原本看似复杂的东西就会变得非常简单!
指针即地址。“地址又是啥?”“只能从汇编语言和计算机组成原理的角度去解释了。”
这辈子不看内存地址和内存值;只画链表、指针示意图,画堆栈示意图,画各种示意图,甚至自己没画过而只看过书上的图……能从本质上理解指针、理解函数参数传递吗?本人深表怀疑!
这辈子不种麦不收麦不将麦粒拿去磨面;只吃馒头、吃面条、吃面包、……甚至从没看过别人怎么蒸馒头,压面条,烤面包,……能从本质上理解面粉、理解面食吗?本人深表怀疑!!
提醒:
“学习用汇编语言写程序”
和
“VC调试(TC或BC用TD调试)时按Alt+8、Alt+6和Alt+5,打开汇编窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应内存和寄存器变化,这样过一遍不就啥都明白了吗。
(Linux或Unix下可以在用GDB调试时,看每句C对应的汇编并单步执行观察相应内存和寄存器变化。)
想要从本质上理解C指针,必须学习C和汇编的对应关系。”
不是一回事!
不要迷信书、考题、老师、回帖;
要迷信CPU、编译器、调试器、运行结果。
并请结合“盲人摸太阳”和“驾船出海时一定只带一个指南针。”加以理解。
任何理论、权威、传说、真理、标准、解释、想象、知识……都比不上摆在眼前的事实!