查看原文
其他

数组和指针一道非常值得深思的笔试题

杨源鑫 嵌入式云IOT技术圈 2021-01-31

之前笔试就遇到下面这道题,谁都不敢说自己的C/C++能有多精通,当然,工作一久,很多老毛病也就容易犯了,所以说,理论是真的很重要的,下面这道题,说实话还是挺基础的,虽然当时笔试被我给猜对了,但还是要深究一下具体的转换细节。

如题:

#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 ;
}

请问以上程序输出结果?程序正确运行结果如下:

我当时一看,数组下标还有负值?这是怎么一回事?我们把上面这个程序变一下,就很清晰了,如下:

#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) ;

我们再来看:

#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 ;
}

运行结果:

如果换种写法,如:

#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 ;
}

你知道答案是多少吗?一样的运算法则:

再接再励!!温故而知新,注重基础,一点细节也不要放过!

往期精彩

数据结构之二叉树

C语言将xxx.bin文件转为数组

别瞎找了,你要的C语言经典示例都在这~

开源STM32产品:无线点菜宝使用评测

STM32CubeMX + STM32F1系列开发时遇到的四个问题及解决方案分享


若觉得本次分享的文章对您有帮助,随手点[在看]并转发分享,也是对我的支持。

    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存