|
12个C语言面试题,涉及指针、进程、运算、结构体、函数、内存,看看你能做出几个!
; X/ `; k0 ?2 u4 B1.gets函数' A' u( R% w3 i
% A" N! L* e: F, V3 |( L( e) T( v问:请找出下面代码里的问题:
& ^: c5 Z, J0 S" M; q ?# ~; L 答:上面代码里的问题在于函数gets的使用,
& t" o$ [8 S j* |2 P" v# ^这个函数从stdin接收一个字符串而不检查它所复制的缓存的容积,; l6 F' h# p! H J; F& {
这可能会导致缓存溢出。这里推荐使用标准函数fgets代替。6 U2 s/ c% V4 g2 y
2.strcpy函数
; K ]0 Z( Z7 G: z; C$ G1 Q Q, }, t- r/ u/ j
问:下面是一个简单的密码保护功能,你能在不知道密码的情况下将其破解吗?
/ \9 O- S) p' U 答:破解上述加密的关键在于利用攻破strcpy函数的漏洞。所以用户在向“passwd”缓存输入随机密码的时候并没有提前检查“passwd”的容量是否足够。所以,如果用户输入一个足够造成缓存溢出并且重写“flag”变量默认值所存在位置的内存的长“密码”,即使这个密码无法通过验证,flag验证位也变成了非零,也就可以获得被保护的数据了。例如:# ]' F# t* Q E- h( l
虽然上面的密码并不正确,但我们仍然可以通过缓存溢出绕开密码安全保护。
, k$ X( Q" T. D要避免这样的问题,建议使用 strncpy函数。
) Z% S% l6 D( D5 Q注:最近的编译器会在内部检测栈溢出的可能,所以这样往栈里存储变量很难出现栈溢出。在我的gcc里默认就是这样,所以我不得不使用编译命令‘-fno-stack-protector’来实现上述方案。) A( ~- X3 t/ Q* u8 N: ^
3.main的返回类型
7 L6 q9 G5 u/ A1 o0 J4 y( ]! }- `6 @4 x- T
问:下面的代码能 编译通过吗?如果能,它有什么潜在的问题吗?3 {5 }- o1 C( Z, p4 G. N; o* `( G
答:因为main方法的返回类型,这段代码的错误在大多数编译器里会被当作警告。main的返回类型应该是“int”而不是“void”。因为“int”返回类型会让程序返回状态值。这点非常重要,特别当程序是作为依赖于程序成功运行的脚本的一部分运行时。! }. n8 K( V2 c* l
4.内存泄露; v$ _! y9 T+ v& g4 g) M4 p
; U2 K; s1 P' t& @; h3 ? C问:下面的代码会导致内存泄漏吗?
& U- `7 B9 w0 m: {- v$ v 答:尽管上面的代码并没有释放分配给“ptr”的内存,但并不会在程序退出后导致内存泄漏。在程序结束后,所有这个程序分配的内存都会自动被处理掉。但如果上面的代码处于一个“while循环”中,那将会导致严重的内存泄漏问题!
$ s4 q" \9 n6 u0 j$ D提示:如果你想知道更多关于内存泄漏的知识和内存泄漏检测工具,可以来看看我们在Valgrind上的文章。1 H- d0 x# z/ g) b z' b' h
5.free函数: j. j# X8 l6 S3 P, J
- u8 U8 @( } \/ R$ K2 @5 g* w7 W
问:下面的程序会在用户输入'freeze'的时候出问题,而'zebra'则不会,为什么?
' N" N7 l6 E5 X. z: J6 B/ k, x 答:这里的问题在于,代码会(通过增加“ptr”)修改while循环里“ptr”存储的地址。当输入“zebra”时,while循环会在执行前被终止,因此传给free的变量就是传给malloc的地址。但在“freeze”时,“ptr”存储的地址会在while循环里被修改,因此导致传给free的地址出错,也就导致了seg-fault或者崩溃。
/ h: T1 N# D x# B1 D2 N8 H6.使用_exit退出
: r$ B/ h3 }8 m: N# |3 p
0 F, g* [% n$ d" O) x |$ _% n, U问:在下面的代码中,atexit并没有被调用,为什么?; o* c& _2 d' \0 A( L7 e0 ~
这是因为_exit函数的使用,该函数并没有调用atexit等函数清理。如果使用atexit就应当使用exit或者“return”与之相配合。9 Q+ o2 k1 u, W- B) `; L) a
7.void*和C结构体
' X" d- M0 [( R
) |3 Y$ `) ?0 L- n1 O5 _问:你能设计一个能接受任何类型的参数并返回interger(整数)结果的函数吗?! I3 _, O k- a/ t$ u$ L* ?
答:如下:
8 D. ?3 G0 z1 I如果这个函数的参数超过一个,那么这个函数应该由一个结构体来调用,这个结构体可以由需要传递参数来填充。! L4 X& L6 w5 h4 [8 k7 b* A5 g7 C
8.*和++操作. _9 y! W" f. j4 P" z, K
' J- O, j% R+ M
问:下面的操作会输出什么?为什么?4 T$ i e- d) a8 x
答:输出结果应该是这样:
+ w; `/ q6 M( P& G; S8 U P' b8 n因为“++”和“*”的优先权一样,所以“*ptr++”相当于“*(ptr++)”。即应该先执行ptr++,然后才是*ptr,所以操作结果是“L”。第二个结果是“i”。
) f- \: l9 @/ p, c: C9.修改代码片段(或者只读代码)
( w, L( k- s% X9 x+ Y1 n' t
, a7 s( P9 L1 v% n问:下面的代码段有错,你能指出来吗?
! ?5 s v* Y% U% \( n4 p 答:这是因为,通过*ptr = ‘T’,会改变内存中代码段(只读代码)“Linux”的第一个字母。这个操作是无效的,因此会造成seg-fault或者崩溃。7 Z( g( l/ m' e# c
10.会改变自己名字的进程
' j$ q* @* J, p& U- H4 S, X4 T* a, v1 W' w" o$ a
问:你能写出一个在运行时改变自己进程名的程序吗?
3 ^1 E8 k5 S/ j5 V) J! c答:参见下面这段代码:
! x' w4 \( o0 P; m4 u 11.返回本地变量的地址! E6 N3 q: ]. s" a$ E6 u
' @$ T% o& V5 X6 K4 V/ o
, o/ ^6 j0 }. a0 ]
问:下面代码有问题吗?如果有,该怎么修改?
. F( f' s- a, T- l0 c8 w6 } 答:尽管上面的程序有时候能够正常运行,但是在“inc”中存在严重的漏洞。这个函数返回本地变量的地址。因为本地变量的生命周期就是“inc”的生命周期,所以在inc结束后,使用本地变量会发生不好的结果。这可以通过将main中变量“a”的地址来避免,这样以后还可以修改这个地址存储的值。1 U- ^ q2 n* M" G2 `5 C9 Z6 r! y! I. K
12.处理printf的参数
/ l2 C# g2 ~- I' Q5 |
+ P' j1 A$ a9 r2 F! O7 u, G问:下面代码会输出什么?
6 \+ Y. S8 V1 o9 J* D 答:输出结果是:& g# m' U3 f9 q5 T2 [/ W# Z# }
. i0 K/ R1 U6 `7 A$ i# y; \: ^. Q$ S. c1 u5 ?' z
这是因为C语言里函数的参数默认是从右往左处理的,输出时是从左往右。9 r" d T7 k8 W5 O$ [ I0 F
另外如果你想学c/c++,笔者这里介绍一个C/C++学习的零基础成长圈子Q羊君,⑤⑥⑨,②⑥⑧,③⑥⑦,要是你对C语言、C++或者是算法方面有兴趣的话,不管你是大牛还是小白,大家都一起成长进步。
. ~3 E3 ?$ E9 V8 m4 M
7 g Q# R0 w2 {1 i% @# N来源:http://www.yidianzixun.com/article/0MaygqsN
2 z) p$ J5 Z6 ?# C$ V: j免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作! |
本帖子中包含更多资源
您需要 登录 才可以下载或查看,没有账号?立即注册
×
|