自各儿看C函数指针的一些困惑
自己看C函数指针的一些困惑
#include<stdio.h>
void MyFun()
{
printf("I'm here!\n");
}
void main()
{
int p=(int)&MyFun;
printf("%d\n",p);
(*(void (*)(void))4198405)();
getchar();
}
请问这个程序为什么在电脑重启,注销什么的都用过之后还是会正确呢(难道是第一次MyFun的函数地址是4198405?)。
我的电脑是32位的,换做64位的还是能正常运行.
这到底是为什么呢?求各位不吝赐教,先行谢过了,谢谢!
还有分不多只有20,聊表谢意
------解决方案--------------------
程序加载到内存的时候,将.EXE加载到进程空间从地址0X400000开始的位置,.EXE的pe头中包含导入表,记录了exe中用到的函数的跳转地址,4198405(0X401005)记录的就是跳转到MyFun函数的JMP 指令。
程序每次运行,都是固定从地址0X400000开始的位置开始加载,导入表中的函数相对位置也都不变,所以4198405始终指向一条跳到MyFun函数的指令,可以加一个MyFun1函数函数,那么MyFun函数在导入表中的相对位置就不是4198405了。
------解决方案--------------------
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、编译器、调试器、运行结果。
并请结合“盲人摸太阳”和“驾船出海时一定只带一个指南针。”加以理解。
任何理论、权威、传说、真理、标准、解释、想象、知识……都比不上摆在眼前的事实!
有人说一套做一套,你相信他说的还是相信他做的?
其实严格来说这个世界上古往今来所有人都是说一套做一套,不是吗?
不要写连自己也预测不了结果的代码!
电脑内存只是一个一维二进制字节数组及其对应的二进制地址;
人脑才将电脑内存中的这个一维二进制字节数组及其对应的二进制地址的某些部分看成是很多数组、指针、数组指针、指针数组、数组的数组、指针的指针、二维数组、……
#include<stdio.h>
void MyFun()
{
printf("I'm here!\n");
}
void main()
{
int p=(int)&MyFun;
printf("%d\n",p);
(*(void (*)(void))4198405)();
getchar();
}
请问这个程序为什么在电脑重启,注销什么的都用过之后还是会正确呢(难道是第一次MyFun的函数地址是4198405?)。
我的电脑是32位的,换做64位的还是能正常运行.
这到底是为什么呢?求各位不吝赐教,先行谢过了,谢谢!
还有分不多只有20,聊表谢意
------解决方案--------------------
程序加载到内存的时候,将.EXE加载到进程空间从地址0X400000开始的位置,.EXE的pe头中包含导入表,记录了exe中用到的函数的跳转地址,4198405(0X401005)记录的就是跳转到MyFun函数的JMP 指令。
程序每次运行,都是固定从地址0X400000开始的位置开始加载,导入表中的函数相对位置也都不变,所以4198405始终指向一条跳到MyFun函数的指令,可以加一个MyFun1函数函数,那么MyFun函数在导入表中的相对位置就不是4198405了。
------解决方案--------------------
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、编译器、调试器、运行结果。
并请结合“盲人摸太阳”和“驾船出海时一定只带一个指南针。”加以理解。
任何理论、权威、传说、真理、标准、解释、想象、知识……都比不上摆在眼前的事实!
有人说一套做一套,你相信他说的还是相信他做的?
其实严格来说这个世界上古往今来所有人都是说一套做一套,不是吗?
不要写连自己也预测不了结果的代码!
电脑内存只是一个一维二进制字节数组及其对应的二进制地址;
人脑才将电脑内存中的这个一维二进制字节数组及其对应的二进制地址的某些部分看成是很多数组、指针、数组指针、指针数组、数组的数组、指针的指针、二维数组、……