首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 开发语言 > C语言 >

printf影响结果?该怎么处理

2012-02-12 
printf影响结果?学习郝斌老师的数据结构,第10讲“跨函数使用内存”中的选择题:题目:通过调用fun函数,main函

printf影响结果?
学习郝斌老师的数据结构,第10讲“跨函数使用内存”中的选择题:
题目:通过调用fun函数,main函数中的指针变量p指向一个合法的整型变量的是:
B.

# include <stdio.h>
int fun(int * * q);
int main(void)
{
int * p;
fun(&p);  
printf("*p = %d\n", *p);
return 0;
}
int fun(int * * q)
{
int s =555;
* q = &s;  
printf("s = %d, **q = %d\n", s, **q); // 这行怎么了?
return **q;
}
/*

fun函数运行完毕,空间释放,*p应该是个随机值,但是——

在VC++6.0中,fun函数中若注释掉printf语句,则输出结果是*p = 555,不注释掉,就是这样的结果:
s = 555, **q = 555
*p = 4198612

用Wint-tc,运行结果跟fun函数中的printf语句没有关系。


有没有printf语句怎么会影响到*p的值?
我不知道这是怎么回事?期望高手解答。

*/

[解决办法]
你可以在fun()里把&s的地址打印出来。
然后在fun()执行之后把&p打印出来,两个地址应该是相等的。
函数调用的时候,&s=0x1234的空间被申请出来。返回之后,编译器不会主动清除刚刚调用的fun()的栈空间。
也就是&s地址保存的值还是555。
如果你打印*p之前,又调用了其他函数。这个地址0x1234就被冲掉了。
[解决办法]
VC调试(TC或BC用TD调试)时按Alt+8、Alt+6和Alt+5,打开汇编窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应内存和寄存器变化,这样过一遍不就啥都明白了吗。
对VC来说,所谓‘调试时’就是编译连接通过以后,按F10或F11键单步执行一步以后的时候,或者在某行按F9设了断点后按F5执行停在该断点处的时候。
(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、编译器、调试器、运行结果。
并请结合“盲人摸太阳”和“驾船出海时一定只带一个指南针。”加以理解。
任何理论、权威、传说、真理、标准、解释、想象、知识……都比不上摆在眼前的事实!

不要写连自己也预测不了结果的代码!

[解决办法]
在 vs2010 中正常
printf("*p = %d\n", *p); // lz 预期得到什么?
  没有注释掉fun函数中的printf语句,则输出结果是*p = 555, 

释掉没有 fun 函数中的 printf 语句,结果都是不可预测的,stack 随着 fun 函数退出而释放,main 中的结果取决于 stack 中的原始数据被覆盖没有,被什么覆盖
[解决办法]
调用别的函数很可能会覆盖。记住,不要访问一个指向无效对象的指针!对象超出生命周期,也是无效的!不该浪费时间研究错误的问题!

热点排行