• 【进阶C语言】数组笔试题解析


    本节内容以刷题为主,大致目录:

    1.一维数组

    2.字符数组

    3.二维数组

    学完后,你将对数组有了更全面的认识


    在刷关于数组的题目前,我们先认识一下数组名:

    数组名的意义:表示数组首元素的地址

    但是有两个例外:

    (1)sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小。单位:字节。

    (2)&数组名,这里的数组名表示整个数组,取出的是整个数组的大小。 

    一、一维数组

    1.判断下列sizeof计算的大小

    (1)一维数组

    1. #include
    2. int main()
    3. {
    4. int a[] = { 1,2,3,4 };//一维数组
    5. printf("%zd\n", sizeof(a));
    6. printf("%zd\n", sizeof(a + 0));
    7. printf("%zd\n", sizeof(*a));
    8. printf("%zd\n", sizeof(a + 1));
    9. printf("%zd\n", sizeof(a[1]));
    10. printf("%zd\n", sizeof(&a));
    11. printf("%zd\n", sizeof(*&a));
    12. printf("%zd\n", sizeof(&a + 1));
    13. printf("%zd\n", sizeof(&a[0]));
    14. printf("%zd\n", sizeof(&a[0] + 1));
    15. return 0;
    16. }

    结果展示:

    代码分析:

    1. #include
    2. int main()
    3. {
    4. int a[] = { 1,2,3,4 };//一维数组
    5. //整形数组,每个元素4个字节,整个数组为16字节
    6. printf("%zd\n", sizeof(a));
    7. //数组名直接放在sizeof内部,表示整个数组的大小
    8. printf("%zd\n", sizeof(a + 0));
    9. //数组名没有单独放在sizeof内部,表示数组首元素的地址。大小就是4/8字节
    10. printf("%zd\n", sizeof(*a));
    11. //a就是首元素地址。*a==*(a+0)==a[0];所以*a就算首元素,大小为4字节
    12. printf("%zd\n", sizeof(a + 1));
    13. //a为首元素地址,+1跳过一个元素的地址,就算第二个元素的地址。a+1==&a[1]
    14. //只要是地址,就算4/8字节
    15. printf("%zd\n", sizeof(a[1]));
    16. //a[1]表示第二个元素,大小就是4字节
    17. printf("%zd\n", sizeof(&a));
    18. //&a,取出的是地址,只要是地址,就是4/8字节
    19. printf("%zd\n", sizeof(*&a));
    20. //*与&操作相互抵消。sizeof(*&a)==sizeof(a),为整个数组元素的大小==16字节
    21. printf("%zd\n", sizeof(&a + 1));
    22. //&a的结果是地址,+1操作后还是地址,那就是4/8字节
    23. printf("%zd\n", sizeof(&a[0]));
    24. //a[0]是第一个元素,&a[0]表示取出第一个元素的地址,地址就算4/8字节
    25. printf("%zd\n", sizeof(&a[0] + 1));
    26. //&a[0]是首元素的地址,&a[0]+1就是第二个元素的地址,大小4/8个字节
    27. return 0;
    28. }

     内存分局图:

    总结:只要可以确定是地址,那么大小一定就是4/8字节

    (2)字符数组

    1. #include
    2. int main()
    3. {
    4. char arr[] = { 'a','b','c','d','e','f' };
    5. printf("%zd\n", sizeof(arr));
    6. printf("%zd\n", sizeof(arr + 0));
    7. printf("%zd\n", sizeof(*arr));
    8. printf("%zd\n", sizeof(arr[1]));
    9. printf("%zd\n", sizeof(&arr));
    10. printf("%zd\n", sizeof(&arr + 1));
    11. printf("%zd\n", sizeof(&arr[0] + 1));
    12. return 0;
    13. }

    结果展示:

    代码分析:

    1. #include
    2. int main()
    3. {
    4. char arr[] = { 'a','b','c','d','e','f' };
    5. //字符数组,每个元素大小1字节,整个数组大小6字节
    6. printf("%zd\n", sizeof(arr));
    7. //计算的是整个数组的大小,为6字节
    8. printf("%zd\n", sizeof(arr + 0));
    9. //计算的是地址,大小为4/8字节
    10. printf("%zd\n", sizeof(*arr));
    11. //*arr为第一个元素,大小为1字节
    12. printf("%zd\n", sizeof(arr[1]));
    13. //arr[1]表示第二个元素,大小为1字节
    14. printf("%zd\n", sizeof(&arr));
    15. //地址,4/8字节
    16. printf("%zd\n", sizeof(&arr + 1));
    17. //地址,4/8字节
    18. printf("%zd\n", sizeof(&arr[0] + 1));
    19. //地址,4/8字节
    20. return 0;
    21. }

    内存布局图:

    (3)字符串数组

    1. #include
    2. int main()
    3. {
    4. char arr[] = "abcdef";
    5. printf("%zd\n", sizeof(arr));
    6. printf("%zd\n", sizeof(arr + 0));
    7. printf("%zd\n", sizeof(*arr));
    8. printf("%zd\n", sizeof(arr[1]));
    9. printf("%zd\n", sizeof(&arr));
    10. printf("%zd\n", sizeof(&arr + 1));
    11. printf("%zd\n", sizeof(&arr[0] + 1));
    12. return 0;
    13. }

    结果展示:

    代码分析:

    1. #include
    2. int main()
    3. {
    4. char arr[] = "abcdef";
    5. //arr中的内容:a,b,c,d,e,f,\0
    6. printf("%zd\n", sizeof(arr));
    7. //整个数组大小:7字节
    8. printf("%zd\n", sizeof(arr + 0));
    9. //地址,4/8字节
    10. printf("%zd\n", sizeof(*arr));
    11. //首元素,1字节大小
    12. printf("%zd\n", sizeof(arr[1]));
    13. //第二个元素,1字节大小
    14. printf("%zd\n", sizeof(&arr));
    15. //地址,4/8字节
    16. printf("%zd\n", sizeof(&arr + 1));
    17. //地址,4/8字节
    18. printf("%zd\n", sizeof(&arr[0] + 1));
    19. //地址。4/8字节
    20. return 0;
    21. }

    内存分布图:

    (4)单单字符串

    1. char *p = "abcdef";
    2. printf("%zd\n", sizeof(p));
    3. printf("%zd\n", sizeof(p+1));
    4. printf("%zd\n", sizeof(*p));
    5. printf("%zd\n", sizeof(p[0]));
    6. printf("%zd\n", sizeof(&p));
    7. printf("%zd\n", sizeof(&p+1));
    8. printf("%zd\n", sizeof(&p[0]+1));

    结果展示:

    代码分析:

    1. #include
    2. int main()
    3. {
    4. char* p = "abcdef";
    5. //p指向a的地址
    6. printf("%zd\n", sizeof(p));
    7. //p是指针,存放首字符a的地址,4/8字节
    8. printf("%zd\n", sizeof(p + 1));
    9. //p+1为字符b的地址,4/8字节
    10. printf("%zd\n", sizeof(*p));
    11. //*p==p[0],拿到的是第一个字符:a,为1字节
    12. printf("%zd\n", sizeof(p[0]));
    13. //求的是字符的大小,1字节
    14. printf("%zd\n", sizeof(&p));
    15. //地址,4/8字节
    16. printf("%zd\n", sizeof(&p + 1));
    17. //地址,4/8字节
    18. printf("%zd\n", sizeof(&p[0] + 1));
    19. //地址,4/8字节
    20. return 0;
    21. }

    内存分布图:

    2.判断下列strlen计算的大小

    知识点:strlen是计算字符串的长度(个数)的函数。统计的是在字符串中\0之前出现的字符的个数

    (1)不带\0的字符数组

    1. #include
    2. #include
    3. int main()
    4. {
    5. char arr[] = { 'a','b','c','d','e','f' };
    6. printf("%d\n", strlen(arr));
    7. printf("%d\n", strlen(arr + 0));
    8. printf("%d\n", strlen(*arr));
    9. printf("%d\n", strlen(arr[1]));
    10. printf("%d\n", strlen(&arr));
    11. printf("%d\n", strlen(&arr + 1));
    12. printf("%d\n", strlen(&arr[0] + 1));
    13. return 0;
    14. }

    因为\0是strlen结束的标志,没有\0则不能正常计算

    所以准确的来说,上述代码都是错误的。

    错误原因解析:

    1. #include
    2. #include
    3. int main()
    4. {
    5. char arr[] = { 'a','b','c','d','e','f' };
    6. printf("%d\n", strlen(arr));
    7. //随机值。地址
    8. printf("%d\n", strlen(arr + 0));
    9. //随机值。地址
    10. printf("%d\n", strlen(*arr));
    11. //非法访问地址,代码错误
    12. printf("%d\n", strlen(arr[1]));
    13. //非法访问地址,错误代码
    14. printf("%d\n", strlen(&arr));
    15. //随机值。地址
    16. printf("%d\n", strlen(&arr + 1));
    17. //随机值。地址
    18. printf("%d\n", strlen(&arr[0] + 1));
    19. //随机值。地址
    20. return 0;
    21. }

    内存分布图: 

    (2)自带\0的字符串数组

    1. #include
    2. #include
    3. int main()
    4. {
    5. char arr[] = "abcdef";
    6. printf("%d\n", strlen(arr));
    7. printf("%d\n", strlen(arr + 0));
    8. printf("%d\n", strlen(*arr));
    9. printf("%d\n", strlen(arr[1]));
    10. printf("%d\n", strlen(&arr));
    11. printf("%d\n", strlen(&arr + 1));
    12. printf("%d\n", strlen(&arr[0] + 1));
    13. return 0;
    14. }

    代码分析:

    1. #include
    2. #include
    3. int main()
    4. {
    5. char arr[] = "abcdef";
    6. //arr中的数据:a,b,c,d,e,f,\0
    7. printf("%d\n", strlen(arr));
    8. //从起始位置开始计算长度,为6
    9. printf("%d\n", strlen(arr + 0));
    10. //从起始位置开始计算长度,为6
    11. printf("%d\n", strlen(*arr));
    12. //*arr==a,传入的值就为97,代码报错
    13. printf("%d\n", strlen(arr[1]));
    14. //arr[1]==b,传入的值为98,代码错误
    15. printf("%d\n", strlen(&arr));
    16. //从起始位置开始计算,长度为6
    17. printf("%d\n", strlen(&arr + 1));
    18. //&arr为起始地址,&arr+1跳过了整个数组,为随机值
    19. printf("%d\n", strlen(&arr[0] + 1));
    20. //从第二个元素开始,为5
    21. return 0;
    22. }

    内存分布图:

    (3)带\0的纯字符串

    1. #include
    2. #include
    3. int main()
    4. {
    5. char* p = "abcdef";
    6. printf("%d\n", strlen(p));
    7. printf("%d\n", strlen(p + 1));
    8. printf("%d\n", strlen(*p));
    9. printf("%d\n", strlen(p[0]));
    10. printf("%d\n", strlen(&p));
    11. printf("%d\n", strlen(&p + 1));
    12. printf("%d\n", strlen(&p[0] + 1));
    13. return 0;
    14. }

    代码分析:

    1. #include
    2. #include
    3. int main()
    4. {
    5. char* p = "abcdef";
    6. printf("%d\n", strlen(p));
    7. //从起始位置开始计算长度,为6
    8. printf("%d\n", strlen(p + 1));
    9. //从b位置开始计算长度,为5
    10. printf("%d\n", strlen(*p));
    11. //*p==a,传入97,代码报错
    12. printf("%d\n", strlen(p[0]));
    13. //代码报错
    14. printf("%d\n", strlen(&p));
    15. //&p为指针变量的地址,为随机值
    16. printf("%d\n", strlen(&p + 1));
    17. //&p+1跳过了该字符串,为随机值
    18. printf("%d\n", strlen(&p[0] + 1));
    19. //从b位置开始计算,长度为5
    20. return 0;
    21. }

    内存分布图:

    二、二维数组

    前言:二维数组的数组名同样是首元素地址,不过二维数组的首元素是第一行元素;而且依旧有两个例外。

    题目:判断下列sizeof计算的大小

    1. #include
    2. int main()
    3. {
    4. int a[3][4] = { 0 };
    5. printf("%zd\n", sizeof(a));
    6. printf("%zd\n", sizeof(a[0][0]));
    7. printf("%zd\n", sizeof(a[0]));
    8. printf("%zd\n", sizeof(a[0] + 1));
    9. printf("%zd\n", sizeof(*(a[0] + 1)));
    10. printf("%zd\n", sizeof(a + 1));
    11. printf("%zd\n", sizeof(*(a + 1)));
    12. printf("%zd\n", sizeof(&a[0] + 1));
    13. printf("%zd\n", sizeof(*(&a[0] + 1)));
    14. printf("%zd\n", sizeof(*a));
    15. printf("%zd\n", sizeof(a[3]));
    16. return 0;
    17. }

    运行结果:

    代码分析:

    1. #include
    2. int main()
    3. {
    4. int a[3][4] = { 0 };
    5. //三行四列,一行的大小是12,整个数组是48
    6. printf("%zd\n", sizeof(a));
    7. //数组名单独放在里面,计算的是整个数组的大小
    8. printf("%zd\n", sizeof(a[0][0]));
    9. //a[0][0]表示第一行第一列的元素,计算的是元素的大小,4字节
    10. printf("%zd\n", sizeof(a[0]));
    11. //a[0]表示第一行的地址,也可以称为第一行这个数组的数组名
    12. //sizeof(arr[0])计算的就是第一行整个一维数组的全部大小,16字节
    13. printf("%zd\n", sizeof(a[0] + 1));
    14. //a[0]没有单独存放,a[0]==&a[0][0],也表示第一行第一个元素的地址
    15. //所以a[0]+1==&a[0][1],地址就是:4/8字节
    16. printf("%zd\n", sizeof(*(a[0] + 1)));
    17. //a[0]+1为第二行元素的数组名,也是第二行元素的首元素,4字节
    18. printf("%zd\n", sizeof(a + 1));
    19. //a没有单独放在sizeof内部,所以表示第一行的地址
    20. //所以a+1是第二行的地址,4/8字节
    21. printf("%zd\n", sizeof(*(a + 1)));
    22. //a+1为第二行的地址,*(a+1)就是第二行
    23. //*(a+1)==a[1],第二行元素的总大小=4*4=16
    24. printf("%zd\n", sizeof(&a[0] + 1));
    25. //&a[0]是第一行的地址,&a[0]+1是第二行的地址
    26. //地址就是4/8字节
    27. printf("%zd\n", sizeof(*(&a[0] + 1)));
    28. //*(&a[0]+1)==a[1],计算的是第二行元素的大小,为16字节
    29. printf("%zd\n", sizeof(*a));
    30. //a没有单独放在sizeof内部,所以是第一行的地址
    31. //*a==a[0],计算的是第一行的元素,为16字节
    32. printf("%zd\n", sizeof(a[3]));
    33. //a[3]表示第四行元素,但是没有第四行元素
    34. //sizeof计算的只是类型大小,a[3]和a[0]或a[1]一样,都表示某一行元素
    35. //虽然没有第四行,但是跟a[0]的类型一样,都是四个int元素的数组
    36. return 0;
    37. }

    内存布局图:

    总结:有二维数组a[3][4]

    (1)a、a[0]、a[1]、a[2]都表示数组名

    (2)a是二维数组的数组名,a[0]、a[1]、a[2]分别是第一、二、三行的数组名。

    (3)第一、第二和第三行数组又可以称为一个一维数组

    (4)a、a[0]、a[1]、a[2]单独放在sizeof内部或者&数组名,就表示整个数组

    (5)没有像(4)那种,a、a[0]、a[1]、a[2]就表示首元素的地址

  • 相关阅读:
    Qt + FFmpeg 搭建 Windows 开发环境
    【性能优化】事件委托
    SRC之若依系统弱口令
    python实现科研通定时自动签到
    leetcode题中的非常见方法归纳
    四大含金量高的算法证书考试
    【排序 - 堆排序】
    maven configuration
    CentOS 7:dmPython安装及测试连接达梦数据库
    罗丹明RB/四甲基罗丹明标记水杨苷Salicin, Rhodamine B/TRITC labeled;Rhodamine B/TRITC-Salicin
  • 原文地址:https://blog.csdn.net/2301_77053417/article/details/133000242