当前位置:   article > 正文

C语言阶段性测试错题纠正与拓展

C语言阶段性测试错题纠正与拓展

引言:在2024年4月26日,我进行了C语言知识“期末考试”。通过这次考试,我发现了我的知识漏洞。所以,我写下这篇博客来记录我的错题,并进行纠正,然后对于以前遗忘知识的回顾

更多有关C语言的知识详解可前往个人主页计信猫

目录

一,选择题

1,第一题

2,第二题

3,第三题

4,第四题

5,第五题

二,编程题

1,第一题

2,第二题

三,遗忘知识的回顾

 1,整型提升和算术提升

Ⅰ,整型提升例题

​编辑 四,勉励自己


一,选择题

1,第一题

 原因分析:在刚看到这道题时,我其实就已经想到了解题的关键,那就是以前所讲到的unsigned char类型的变量的取值范围,如下图:

        可是,就在我进行对i值的不断减三时,我好似发现了一个可以秒杀这道题的地方,那就是unsigned char无符号整型,那这个值i就永远不可能为负数对吧,那退出for循环的条件肯定永远都不会被满足啊,那么不直接就死循环了吗?那答案就是B啊。

        所以这道题我就败在了看题的准确度上,仔细读题,才发现for循环的判断条件是i>0,那么也就意味着,i=0时也可以退出循环

 题目正解:正如前面所提到的unsigned char类型的取值范围只可以为0~255,所以i初始化为7,那么对其不断地减去三,i的值则会如下变化:

        那么每减去一次三,就++j,那么具体的减三次数如下图所示:

        那么2+1+84+1+85=173次,所以答案应该选C。 

2,第二题

原因分析:说实话,这道题错了真该挨巴掌。我当时其实也是分析出来了,x首先为1,经过printf语句后,打印x值为1,之后x加一为2,通过while循环判断条件后x又减一为1,如此往复循环,那么应该是陷入了死循环。但是我当时脑子一抽,认为A选项的1表示的就是死循环的打印1,只是太长了,就没有全部写出来。现在一回想真的是后悔莫及啊!

题目正解:直接上图,不废话了:

        所以如图所示,1,2过程一直循环,始终跳不出while循环,故死循环打印1,答案选D。 

3,第三题

原因分析: 这道题我错的原因是知识点掌握不牢固,感觉每一句话都有模有样很有道理,最后迫不得已扣了字眼,认为D选项不对,一个普普通通的#define的宏定义这么简单的写法,怎么可能比函数(如此强大的工具)效率更高?而事实证明我还是错了,计算机语言做题是十分严谨的,不能去抠字眼,每一道题的答案都要选的有根有据

题目正解:这道题虽然涉及到了C++的语法知识,但是通过排除法还是可以做出来的。从我的笔记来看,我所能归纳出的#define宏的优缺点如下

优点:

1,宏的规模和速度更胜于函数一筹

2,宏的类型与参数的类型无关

缺点:

1,宏在替换时会延长代码

2,宏无法通过调试判断是否出错

3,宏无类型,不严谨

4,宏会带来运算符优先级的问题

        所以根据如上归纳,选项ACD都是对的,那么排除法就是选B。 

4,第四题

 原因分析:这道题的错误原因是知识点掌握错误,而导致了这道题的错误,我的错误笔记如下:

        做这道题时,我还保留着错误的认知,认为int*为4个字节,char*为1个字节,那么不就大小不同吗?所以我就选择了错误答案B(现在错误的笔记已被改正) 

题目正解:我们直接在x86和x64的环境下进行代码实操就行了:

  1. #include <stdio.h>
  2. int main()
  3. {
  4. printf("%zd\n", sizeof(int*));
  5. printf("%zd\n", sizeof(char*));
  6. printf("%zd\n", sizeof(float*));
  7. printf("%zd\n", sizeof(double*));
  8. printf("%zd\n", sizeof(short*));
  9. printf("%zd\n", sizeof(long*));
  10. return 0;
  11. }

x86环境:

x64环境:

        可以看出,指针变量的大小只与平台有关,32位机为4,64位机则为8。而这道题真正错误的是A选项当我们使用free函数释放掉某个指针所指向的内容时,该指针并不会自动置为NULL,必须我们手动置空。 

5,第五题

原因分析: 这道题错误的原因也是我的知识点掌握不牢固,别无其他原因。

题目正解:我将如下归纳预处理,编译,链接,执行的作用:

一,预处理:展开宏定义,展开头文件,甚至进行更多预处理指令操作(如#error #pragma #line等预处理指令)。

二,编译:将预处理后的⽂件进⾏⼀系列的:词法分析、语法分析、语义分析及优化,发现语法错误,⽣成相应的汇编代码⽂件。

三,链接:地址和空间分配,符号决议和重定位等这些步骤,链接解决的是⼀个项⽬中多⽂件、多模块之间互相调⽤的问题(于此中发现函数未被定义)。

四,执行:不会引发调试器或测试框架,只是简单地启动应用程序并按照程序代码的逻辑执行代码。

二,编程题

1,第一题

题目正解:首先,我们可以使用以前学到的%10来获取数字的最后一位,/10来去掉数字的最后一位,然后,再将所得到的数字以字符的形式储存到字符数组中,并且每隔三个字符,就额外储存一个“,”,最后,我们再将字符串打印出来就可以了。那么代码如下:

  1. #include <stdio.h>
  2. int main()
  3. {
  4. int input = 0;
  5. scanf("%d", &input);
  6. char arr[100] = { 0 };
  7. int save = 0;
  8. int i = 0;
  9. int count = 0;
  10. while (input)
  11. {
  12. if (count != 0 && count % 3 == 0)
  13. {
  14. arr[i++] = ',';
  15. }
  16. save = input % 10;//获取数字的最后一位
  17. input /= 10;//删除数字的最后一位
  18. arr[i++] = save + '0';//将得到的数字以字符形式储存
  19. count++;
  20. }
  21. i -= 1;//在while循环最后一次时,i++使i多了个1,所以要将i减1
  22. printf("%s", arr);
  23. return 0;
  24. }

2,第二题

题目正解:对于这道题目,我们的思路就是首先创造两个字符数组,我们将第一个字符数组的每一个字符于第二个字符数组中遍历,如果第二个字符数组中有该字符,则不打印;反之,就打印该字符所以代码如下:

  1. #include<stdio.h>
  2. int is_exist(char ch, char arr[])
  3. {
  4. int i = 0;
  5. while (arr[i])
  6. {
  7. if (arr[i] == ch)
  8. {
  9. //在第二个字符串中
  10. return 0;
  11. }
  12. i++;
  13. }
  14. //不在第二个字符串中
  15. return 1;
  16. }
  17. int main()
  18. {
  19. char arr1[101] = { 0 };
  20. char arr2[101] = { 0 };
  21. //获得字符串放入两个字符数组
  22. gets(arr1);
  23. gets(arr2);
  24. int i = 0;
  25. while (arr1[i])
  26. {
  27. if (is_exist(arr1[i], arr2))
  28. {
  29. printf("%c", arr1[i]);
  30. }
  31. i++;
  32. }
  33. return 0;
  34. }

三,遗忘知识的回顾

 1,整型提升和算术提升

        在考试中,曾有这么一道题:

        而这道题中,就涉及到了整型提升和算术提升的知识点。整型提升:当变量类型的级别低于int时,算数运算时就会将变量的类型提升为int。算术提升:当变量类型的级别高于int时,算数运算时就会将变量的类型提升为更高的类型

        如此,答案为D选项

Ⅰ,整型提升例题

        那我们现在来看一道与整型提升有关的例题:

  1. #include<stdio.h>
  2. int main()
  3. {
  4. char a = 5;
  5. char b = 28;
  6. char sum = a + b;
  7. printf("%d\n", sum);
  8. printf("%c\n", sum);
  9. return 0;
  10. }

        那么这段代码的输出结果是什么呢?

题目解答: 这道题之中,我们将数字228分别赋给字符变量ab,后面再执行a+b并将结果赋值给sum。那么很明显,在相加的时候,就会发生前面是所提到的整型提升

        所以sum=a+b33,再将33对应的阿斯克码值换算出为'!',故打印出的结果如下:

小问题的解答: 在做这道题时,其实我就会想到一个问题,我们使用字符类型来定义变量a,b和sum,为什么我们却使用整型变量来初始化它们呢?这不就乱套了吗?

        其实,这样初始化并没有错误。因为在计算机中,任何数据都是以二进制代码进行储存的,而每一个字符都有自己对应的阿斯克码值阿斯克码值不就是整型数字吗?那么我们就可以理解为其实字符变量类型储存的其实也是数字,所以我们拿整型对某个字符变量初始化并无不妥

        请注意!如下两种初始化是有区别的!

  1. #include<stdio.h>
  2. int main()
  3. {
  4. char a = 0;//将数字0初始化a,0的阿斯克码值所对应的字符为NULL(无法被打印)
  5. char b = '0';//将字符0初始化b
  6. printf("%c\n", a);
  7. printf("%c\n", b);
  8. return 0;
  9. }

        代码一走,结果如下:

 四,勉励自己

        虽然这次考试成绩不理想,但是经过这次的写博客进行总结,感觉学到了很多东西,越来越上道了

        虽然这篇博客的技术含量不高,但是对我来说意义重大这篇博客不仅让我搞懂了以前没有搞懂的知识,并且还让我有了以前上高中时,对着自己没考好的数学卷子改错纠错记错题的感觉,这种记错题的好习惯真的让我受益匪浅。加油吧,每天学习一点点,百万年薪正等着我呢!!!

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/很楠不爱3/article/detail/535218
推荐阅读
相关标签
  

闽ICP备14008679号