12个有趣的C语言面试题
摘要:12个C语言面试题,涉及指针、进程、运算、结构体、函数、内存,看看你能做出几个!
1.gets()函数
问:请找出下面代码里的问题:
- #include<stdio.h>
 - int main(void)
 - {
 - char buff[10];
 - memset(buff,0,sizeof(buff));
 - gets(buff);
 - printf("\n The buffer entered is [%s]\n",buff);
 - return 0;
 - }
 
答:上面代码里的问题在于函数gets()的使用,这个函数从stdin接收一个字符串而不检查它所复制的缓存的容积,这可能会导致缓存溢出。这里推荐使用标准函数fgets()代替。
2.strcpy()函数
问:下面是一个简单的密码保护功能,你能在不知道密码的情况下将其破解吗?
- #include<stdio.h>
 - int main(int argc, char *argv[])
 - {
 - int flag = 0;
 - char passwd[10];
 - memset(passwd,0,sizeof(passwd));
 - strcpy(passwd, argv[1]);
 - if(0 == strcmp("LinuxGeek", passwd))
 - {
 - flag = 1;
 - }
 - if(flag)
 - {
 - printf("\n Password cracked \n");
 - }
 - else
 - {
 - printf("\n Incorrect passwd \n");
 - }
 - return 0;
 - }
 
答:破解上述加密的关键在于利用攻破strcpy()函数的漏洞。所以用户在向“passwd”缓存输入随机密码的时候并没有提前检查“passwd”的容量是否足够。所以,如果用户输入一个足够造成缓存溢出并且重写“flag”变量默认值所存在位置的内存的长“密码”,即使这个密码无法通过验证,flag验证位也变成了非零,也就可以获得被保护的数据了。例如:
- $ ./psswd aaaaaaaaaaaaa
 - Password cracked
 
虽然上面的密码并不正确,但我们仍然可以通过缓存溢出绕开密码安全保护。
要避免这样的问题,建议使用 strncpy()函数。
作者注:最近的编译器会在内部检测栈溢出的可能,所以这样往栈里存储变量很难出现栈溢出。在我的gcc里默认就是这样,所以我不得不使用编译命令‘-fno-stack-protector’来实现上述方案。
3.main()的返回类型
问:下面的代码能 编译通过吗?如果能,它有什么潜在的问题吗?
- #include<stdio.h>
 - void main(void)
 - {
 - char *ptr = (char*)malloc(10);
 - if(NULL == ptr)
 - {
 - printf("\n Malloc failed \n");
 - return;
 - }
 - else
 - {
 - // Do some processing
 - free(ptr);
 - }
 - return;
 - }
 
答:因为main()方法的返回类型,这段代码的错误在大多数编译器里会被当作警告。main()的返回类型应该是“int”而不是“void”。因为“int”返回类型会让程序返回状态值。这点非常重要,特别当程序是作为依赖于程序成功运行的脚本的一部分运行时。
4.内存泄露
问:下面的代码会导致内存泄漏吗?
- #include<stdio.h>
 - void main(void)
 - {
 - char *ptr = (char*)malloc(10);
 - if(NULL == ptr)
 - {
 - printf("\n Malloc failed \n");
 - return;
 - }
 - else
 - {
 - // Do some processing
 - }
 - return;
 - }
 
答:尽管上面的代码并没有释放分配给“ptr”的内存,但并不会在程序退出后导致内存泄漏。在程序结束后,所有这个程序分配的内存都会自动被处理掉。但如果上面的代码处于一个“while循环”中,那将会导致严重的内存泄漏问题!
提示:如果你想知道更多关于内存泄漏的知识和内存泄漏检测工具,可以来看看我们在Valgrind上的文章。
5.free()函数
问:下面的程序会在用户输入'freeze'的时候出问题,而'zebra'则不会,为什么?
- #include<stdio.h>
 - int main(int argc, char *argv[])
 - {
 - char *ptr = (char*)malloc(10);
 - if(NULL == ptr)
 - {
 - printf("\n Malloc failed \n");
 - return -1;
 - }
 - else if(argc == 1)
 - {
 - printf("\n Usage \n");
 - }
 - else
 - {
 - memset(ptr, 0, 10);
 - strncpy(ptr, argv[1], 9);
 - while(*ptr != 'z')
 - {
 - if(*ptr == '')
 - break;
 - else
 - ptr++;
 - }
 - if(*ptr == 'z')
 - {
 - printf("\n String contains 'z'\n");
 - // Do some more processing
 - }
 - free(ptr);
 - }
 - return 0;
 - }
 
答:这里的问题在于,代码会(通过增加“ptr”)修改while循环里“ptr”存储的地址。当输入“zebra”时,while循环会在执行前被终止,因此传给free()的变量就是传给malloc()的地址。但在“freeze”时,“ptr”存储的地址会在while循环里被修改,因此导致传给free()的地址出错,也就导致了seg-fault或者崩溃。
6.使用_exit退出
问:在下面的代码中,atexit()并没有被调用,为什么?
- #include<stdio.h>
 - void func(void)
 - {
 - printf("\n Cleanup function called \n");
 - return;
 - }
 - int main(void)
 - {
 - int i = 0;
 - atexit(func);
 - for(;i<0xffffff;i++);
 - _exit(0);
 - }
 
7.void*和C结构体
问:你能设计一个能接受任何类型的参数并返回interger(整数)结果的函数吗?
答:如下:
- int func(void *ptr)
 
如果这个函数的参数超过一个,那么这个函数应该由一个结构体来调用,这个结构体可以由需要传递参数来填充。
8.*和++操作
问:下面的操作会输出什么?为什么?
- #include<stdio.h>
 - int main(void)
 - {
 - char *ptr = "Linux";
 - printf("\n [%c] \n",*ptr++);
 - printf("\n [%c] \n",*ptr);
 - return 0;
 - }
 
答:输出结果应该是这样:
- [L]
 - [i]
 
因为“++”和“*”的优先权一样,所以“*ptr++”相当于“*(ptr++)”。即应该先执行ptr++,然后才是*ptr,所以操作结果是“L”。第二个结果是“i”。
9.问:修改代码片段(或者只读代码)
问:下面的代码段有错,你能指出来吗?
- #include<stdio.h>
 - int main(void)
 - {
 - char *ptr = "Linux";
 - *ptr = 'T';
 - printf("\n [%s] \n", ptr);
 - return 0;
 - }
 
答:这是因为,通过*ptr = ‘T’,会改变内存中代码段(只读代码)“Linux”的第一个字母。这个操作是无效的,因此会造成seg-fault或者崩溃。
10.会改变自己名字的进程
问:你能写出一个在运行时改变自己进程名的程序吗?
答:参见下面这段代码:
- #include<stdio.h>
 - int main(int argc, char *argv[])
 - {
 - int i = 0;
 - char buff[100];
 - memset(buff,0,sizeof(buff));
 - strncpy(buff, argv[0], sizeof(buff));
 - memset(argv[0],0,strlen(buff));
 - strncpy(argv[0], "NewName", 7);
 - // Simulate a wait. Check the process
 - // name at this point.
 - for(;i<0xffffffff;i++);
 - return 0;
 - }
 
11.返回本地变量的地址
问:下面代码有问题吗?如果有,该怎么修改?
- #include<stdio.h>
 - int* inc(int val)
 - {
 - int a = val;
 - a++;
 - return &a;
 - }
 - int main(void)
 - {
 - int a = 10;
 - int *val = inc(a);
 - printf("\n Incremented value is equal to [%d] \n", *val);
 - return 0;
 - }
 
答:尽管上面的程序有时候能够正常运行,但是在“inc()”中存在严重的漏洞。这个函数返回本地变量的地址。因为本地变量的生命周期就是“inc()”的生命周期,所以在inc结束后,使用本地变量会发生不好的结果。这可以通过将main()中变量“a”的地址来避免,这样以后还可以修改这个地址存储的值。
12.处理printf()的参数
问:下面代码会输出什么?
- #include<stdio.h>
 - int main(void)
 - {
 - int a = 10, b = 20, c = 30;
 - printf("\n %d..%d..%d \n", a+b+c, (b = b*2), (c = c*2));
 - return 0;
 - }
 
答:输出结果是:
- 110..40..60
 
这是因为C语言里函数的参数默认是从右往左处理的,输出时是从左往右。
12个有趣的C语言面试题的更多相关文章
- 12个有趣的c语言面试题 
		
1.gets()函数 问:请找出下面代码里的问题: #include int main(void) { char buff[10]; memset(buff,0,sizeof(buff)); gets ...
 - 12个有趣的C语言问答(详解)
		
本文参照博文<12个有趣的C语言问答>,在原文的基础上增加来对应的知识点的详细介绍. 1 gets()方法 Q:下面的代码有一个被隐藏的问题,你能找到它吗? #include <st ...
 - 《12个有趣的C语言问答》(4)
		
C语言面试问答——<12个有趣的C语言问答>评析(4) 前文链接:http://www.cnblogs.com/pmer/p/3324063.html 8,Making changes i ...
 - 《12个有趣的C语言问答》评析2
		
<12个有趣的C语言问答>评析(2) 前文链接:http://www.cnblogs.com/pmer/p/3313913.html (没存盘,遭遇过热保护.至少4个问答的评论白写了.默哀 ...
 - 12个滑稽的C语言面试问答——《12个有趣的C语言问答》评析(5)
		
前文链接:http://www.cnblogs.com/pmer/archive/2013/09/17/3327262.html A,局部变量的返回地址 Q:下面的代码有问题吗?如果有,如何修改? # ...
 - 12个有趣的C语言问答
		
转自:http://www.admin10000.com/document/913.html 1,gets() 方法 Q:以下代码有个被隐藏住的问题,你能找到它吗? 1 2 3 4 5 6 7 8 9 ...
 - C语言面试题(嵌入式开发方向,附答案及点评)
		
整理自C语言面试题(嵌入式开发方向,附答案及点评) 预处理器(Preprocessor) 1. 用预处理指令#define 声明一个常数,用以表明1年中有多少秒(忽略闰年问题) #define SEC ...
 - 嵌入式开发—C语言面试题
		
嵌入式开发—C语言面试题 源地址:http://blog.csdn.net/xdx2ct1314/article/details/7358929 1. 用预处理指令#define 声明一个常数,用 ...
 - C语言面试题汇总之一
		
C语言面试题汇总之一 1.static有什么用途?(请至少说明两种) l 限制变量的作用域: 设置变量的存储域. 2.引用和指针有什么区别? 引用必须被初始化,指针不必: 引用初始化以后不能被改变,指 ...
 
随机推荐
- HighChart图片本地导出
			
Highchart第三方图表控件,导出默认是从官方地址导出,这样在无外网的条件下则导致导出失败,改进如下: 后台导出代码: public partial class HighChart : Syste ...
 - Firefox和Chrome浏览器导出书签
			
Chrome浏览器: 或者直接在地址栏中输入:“chrome://bookmarks/#1”也可以 Firefox浏览器:
 - 对象不支持“attachEvent”属性或方法的解决办法
			
有些脚本在IE11下执行会报错误: 对象不支持“attachEvent”属性或方法 解决办法 解决办法:把attachEvent改为addEventListener即可
 - HDU 1686 (KMP模式串出现的次数) Oulipo
			
题意: 求模式串W在母串T中出现的次数,各个匹配串中允许有重叠的部分. 分析: 一开始想不清楚当一次匹配完成时该怎么办,我还SB地让i回溯到某个位置上去. 后来仔细想想,完全不用,直接让模式串向前滑动 ...
 - POJ 3041 Asteroids (最小点覆盖集)
			
题意 给出一个N*N的矩阵,有些格子上有障碍,要求每次消除一行或者一列的障碍,最少消除多少次可以全部清除障碍. 思路 把关键点取出来:一个障碍至少需要被它的行或者列中的一个消除. 也许是最近在做二分图 ...
 - Control File (二)重建CONTROLFILE  --- NORESETLOG
			
create controlfile --- noresetlog 由于丢失control01.ctl alter_karl.log 中显示: -------------------------- ...
 - ORACLE之手动注册监听listener。alter system set local_listener="XXX"
			
记录下刚刚做的一个为一个数据库(t02)配置多个监听(listener)的实验,过程有点小曲折. (1)新增两个测试的监听,listener.ora的配置内容(可纯手动编辑该文件或使用netca)如下 ...
 - Oracle 不同故障的恢复方案
			
之前在Blog中对RMAN 的备份和恢复做了说明,刚看了下,在恢复这块还有知识点遗漏了. 而且恢复这块很重要,如果DB 真要出了什么问题,就要掌握对应的恢复方法. 所以把DB的恢复这块单独拿出来说明一 ...
 - BasicDataSource配备
			
BasicDataSource配置 commons DBCP 配置参数简要说明 前段时间因为项目原因,要在修改数据库连接池到DBCP上,折腾了半天,有一点收获,不敢藏私,特在这里与朋友们共享. 在配置 ...
 - 底部菜单栏(一) TabHost实现
			
需求:使用TabHost实现底部菜单栏: 效果图: 实现分析: 1.目录结构: 代码实现: 1.activity_main.xml <?xml version="1.0" e ...