五行代码的疑惑(关于指针的深刻理解...)2007/08/12议题作者:sudami信息:黑客基地(bbs.hackbase)文章内容:有人曾问过我一道C语言的题目,如下#includevoidpo(char*pt){pt+=3;}voidmain(){char*pt,str[]="aceg";pt=str;po(pt);cout<<*pt;}除掉函数头和花括号之后就只有五行,这个关于指针操作的程序看似简单,我想大部分人看到这个题目,都会毫不犹豫的说输出"g",不知道计算机前的你会怎样认为。我曾拿她问过几位比较有经验的程序员,他们也是这样认为。我先不想对结果发表看法,只想问一下你怎样看传值调用与传址调用。很显然这个题目是关于传址的,也正是因为这点,才会有人这么肯定"g"的答案。如果你仔细想一想的话,你又怎么认为呢?该怎样改才算正确呢?有几种改法呢?你又联想到了哪些呢?--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------正确结果:那么我来分析下:voidpo(char*pt){pt+=3;}参数是一个指针,这个指针本身是一个局部变量(分配在栈上),它得到的值是外部代码传给它的一个副本值,而不是副本变量本身的地址。所以当它的值变化时,外界的那个指针的值是不会变化的。将指针值pt作为值进栈,op里虽然改变了,出栈依旧是原值.po那个函数没有真正改变指针pt的位置。所以,看到的,还是字符串的第一个字母a.通常有两种改法:voidpo(char**pt){}//C写法(就是用个二级指针来存放main里pt的地址!这样改变po里的pt就会改变main里pt的指向了)或者voidpo(char*pt){}//C++写法一句话:子函数需要改变父函数的某个值需要给子函数传递指针进一步深刻的分析,看下面3个例子:①#includeintpo(char*pt){*pt=*pt+3;return(*pt);}voidmain(){char*pt,str[]="aceg";pt=str;po(pt);cout<<*pt;}结果是d!因为PO在得到了pt指针所指向的str的0地址后,是把str[0]的值+3后返回的,就是a+3,这是ASC值在增加,结果当然是d.②#includeintpo(char*pt){*pt=*(pt+3);return(*pt);}voidmain(){char*pt,str[]="aceg";pt=str;po(pt);cout<<*pt;}结果是g!!因为PO在得到了pt指针所指向的str的0地址后,是把这个0指针+3后返回的,就是str[0+3]=str[3],结果当然是g.③#includeintpo(char*pt){pt=pt+3;/*看清楚了,这里的pt只是函数po的局部变量,它和主函数中的那个pt是完全不同的,也就是他在这里对主函数中的操作没有任何作用,因为它不会返回到主函数中执行*/return(*pt);}voidmain(){char*pt,str[]="aceg";pt=str;po(pt);cout<<*pt;}结果是a!!这个要非常注意!其实从变量的作用域的角度可以这样解释:在main函数中定义的指针变量pt的作用域只局限于main函数内,而po函数中的形参指针变量pt的作用域也只在po函数内。因此在main函数中”cout<<*pt;"中的pt是main函数中的pt。所以输出结果还是“abcd”.在main里,指针变量pt是存着str的首地址的,传给po时也是传str的首地址过去po里的pt!但两个pt都只是在各自的函数里的变量(局部变量)!po里的pt影响不了main里的pt如果你把输出语句方到po函数体内的话,则是po函数中的pt了.如下:#includeintpo(char*pt){pt=pt+3;/*看清楚了,这里的pt只是函数po的局部变量,它和主函数中的那个pt是完全不同的,也就是他在这里对主函数中的操作没有任何作用,因为它不会返回到主函数中执行*/cout<<*pt<