最近笔试就遇到下面这道题,谁都不敢说自己的C/C++能有多精通,当然,工作一久,很多老毛病也就容易犯了,所以说,理论是真的很重要的,下面这道题,说实话还是挺基础的,虽然当时笔试被我给猜对了,但还是要深究一下具体的转换细节。
如题:
1
2
3
4
5
6
7
8
9
|
#include <stdio.h> int main( void ) { char *str[] = { "ab" , "cd" , "ef" , "gh" , "ij" , "kl" }; char *t ; t = (str+4)[-1]; printf ( "%s\n" ,t); return 0 ; } |
请问以上程序输出结果?程序正确运行结果如下:
我当时一看,数组下标还有负值?这是怎么一回事?我们把上面这个程序变一下,就很清晰了,如下:
1
2
3
4
5
6
7
8
9
10
11
|
#include <stdio.h> int main( void ) { char *str[] = { "ab" , "cd" , "ef" , "gh" , "ij" , "kl" }; char *t ; // t = (str+4)[-1]; // printf("%s\n",t); t = (str+4)[0] ; printf ( "t:%s\n" ,t); return 0 ; } |
这个程序毫无疑问,答案就是ij。
看上面这幅图即可得到结果,其实就是这么一个转换关系:
实际上编译系统将数组元素的形式a[i]转换成*(a+i),然后才进行运算。对于一般数组元素的形式: <数组名>[<下标表达式>] 编译程序将其转换成:*(<数组名>+<下标表达式>),其中下标表达式为:下标表达式*扩大因子。整个式子计算结果是一个内存地址,最后的结果为:*<地址>=<地址所对应单元的地址的内容>。由此可见,C语言对数组的处理,实际上是转换成指针地址的运算。
所以,上面的式子的转换结果就是:t = *(str+4);
所以,t = (str+4)[-1] =======> t = *(str+4-1) ======> t = *(str+3) ;
所以:
1
2
3
4
5
6
7
8
9
10
11
|
#include <stdio.h> int main( void ) { char *str[] = { "ab" , "cd" , "ef" , "gh" , "ij" , "kl" }; char *t ; // t = (str+4)[-1]; // printf("%s\n",t); t = *(str+4-1); printf ( "t:%s\n" ,t); return 0 ; } |
运行结果:
如果换种写法,如:
1
2
3
4
5
6
7
8
9
10
|
#include <stdio.h> int main( void ) { int b ; int a[10] = {1,2,3,4,5,6,7,8,9,10}; int *p = &a[0] ; b = (p+8)[-4]; printf ( "b:%d\n" ,b); return 0 ; } |
你能知道答案是多少吗?一样的运算法则:
再接再励!!温故而知新,注重基础,一点细节也不要放过!
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对服务器之家的支持。如果你想了解更多相关内容请查看下面相关链接
原文链接:https://blog.csdn.net/morixinguan/article/details/78885010