两年前找工作笔试遇到下面这道题,说实话还是挺基础的,虽然当时笔试被我给猜对了,但还是要深究一下具体的转换细节。 题目如下:
1#include <stdio.h>
2int main(void)
3{
4 char *str[] = {"ab","cd","ef","gh","ij","kl"};
5 char *t ;
6 t = (str+4)[-1];
7 printf("%s\n",t);
8 return 0 ;
9}
请问以上程序输出结果?
程序正确运行结果如下,请问为什么?
我当时一看,数组下标还有负值?这是怎么一回事?我们把上面这个程序变一下,就很清晰了,如下:
1#include <stdio.h>
2int main(void)
3{
4 char *str[] = {"ab","cd","ef","gh","ij","kl"};
5 char *t ;
6// t = (str+4)[-1];
7// printf("%s\n",t);
8 t = (str+4)[0] ;
9 printf("t:%s\n",t);
10 return 0 ;
11}
这个程序毫无疑问,答案就是ij。
看上面这幅图即可得到结果,其实就是这么一个转换关系。实际上编译系统将数组元素的形式a[i]转换成*(a+i),然后才进行运算。对于一般数组元素的形式: <数组名>[<下标表达式>] 编译程序将其转换成:
*(<数组名>+<下标表达式>)。其中下标表达式为:下标表达式*扩大因子。整个式子计算结果是一个内存地址,最后的结果为:*<地址>=<地址所对应单元的地址的内容>。由此可见,C语言对数组的处理,实际上是转换成指针地址的运算。
所以,上面的式子的转换结果就是:t = *(str+4);所以由 t = (str+4)[-1] =======> t = *(str+4-1) ======> t = *(str+3) ; 所以代码如下:
1#include <stdio.h>
2int main(void)
3{
4 char *str[] = {"ab","cd","ef","gh","ij","kl"};
5 char *t ;
6// t = (str+4)[-1];
7// printf("%s\n",t);
8 t = *(str+4-1);
9 printf("t:%s\n",t);
10 return 0 ;
11}
运行结果:
如果换种写法,如:
1#include <stdio.h>
2int main(void)
3{
4 int b ;
5 int a[10] = {1,2,3,4,5,6,7,8,9,10};
6 int *p = &a[0] ;
7 b = (p+8)[-4];
8 printf("b:%d\n",b);
9 return 0 ;
10}
你能知道答案是多少吗?一样的运算法则:
再接再励!!温故而知新,注重基础,一点细节也不要放过!