C/C++基础面试试题大全

第一章 面试基础

  •   一年一度招聘开始了,又有很多人可能像我当初一样,开始搜寻面试题,
     故本人将自己的学习经验(c面试题库整理)进行分享,希望可以帮助到看到的同行!
     
     简要说明:本文第一章是一些C/C++的一些面试基础题库,第二章是一些从网
     络、同事、个人的一些处理简历的技巧
     第三章是将搜集到一些HR查看简历的经验,进行分享。
     													   时间 2020.09.01  作者:ctb
    

一、基础部分

1.关键字

试题1:关键字const有什么含意?
	  答案:
     (1)可以修饰const 常变量
	 (2const可以修饰函数的参数、返回值,甚至函数的定义体。被const修饰的东西	
	 都受到强制
试题2:分析以下代码定义,说明其特性	
	 const int a;
	 int const a;
	 const int *a;
	 int * const a;
	 int const * a const;

答案:	前两个的作用是一样,a是一个常整型数。
	第三个意味着a是一个指向常整型数的指针(也就是,整型数是不可修改的,但指针可以)。
	
	第四个意思a是一个指向整型数的常指针(也就是说,指针指向的整型数是可以修改的,但指
	针是不可修改的)。
	最后一个意味着a是一个指向常整型数的常指针(也就是说,指针指向的整型数是不可修改
	的,同时指针也是不可修改的)。
  思考3:const修饰的常量与宏定义常量的区别
答案:
	const 常量有数据类型,而宏常量没有数据类型。编译器可以对前者进行类型安全检查。
	而对后者只进行字符替换,没有类型安全检查,并且在字符替换可能会产生意料不到的错
	误。
试题4: 关键字volatile有什么含意?并给出三个不同的例子。
答案:
	一个定义为volatile的变量是说这变量可能会被意想不到地改变,这样,编译器就不会去假
	设这个变量的值了。
	精确地说就是,优化器在用到这个变量时必须每次都小心地重新读取这个变量的值,而不是
	使用保存在寄存器里的备份。
	1) 并行设备的硬件寄存器(如:状态寄存器)
	2) 一个中断服务子程序中会访问到的非自动变量(Non-automatic variables)
	3) 多线程应用中被几个任务共享的变量保护,可以预防意外的变动,能提高程序的健壮性。
试题51)一个参数既可以是const还可以是volatile吗?解释为什么。
	2); 一个指针可以是volatile 吗?解释为什么。
	3); 下面的函数有什么错误:
	int square(volatile int *ptr)
	{
   
			return *ptr * *ptr;
	}

答案:
	1)是的。一个例子是只读的状态寄存器。它是volatile因为它可能被意想不到地改变。它是
	const因为程序不应该试图去修改它。
	2)是的。尽管这并不很常见。一个例子是当一个中服务子程序修该一个指向一个buffer的指
	针时。
	3) 这段代码有点变态。这段代码的目的是用来返指针*ptr指向值的平方,但是,由于*ptr
	指向一个volatile型参数,编译器将产生类似下面的代码:
	int square(volatile int *ptr) 
	{
   
		int a,b;
		a = *ptr;
		b = *ptr;
		return a * b;
	}
	由于*ptr的值可能被意想不到地该变,因此a和b可能是不同的。结果,这段代码可能返不是
	你所期望的平方值!正确的代码如下:
	long square(volatile int *ptr) 
	{
   
		int a;
		a = *ptr;
		return a * a;
	}		
试题6. 关键字static的作用是什么?
答案:
	在C语言中,关键字static有三个明显的作用:
1)在函数体,一个被声明为静态的变量在这一函数被调用过程中维持其值不变。
2) 在模块内(但在函数体外),一个被声明为静态的变量可以被模块内所用函数访问,但不能被
模块外其它函数访问。它是一个本地的全局变量。
3) 在模块内,一个被声明为静态的函数只可被这一模块内的其它函数调用。那就是,这个函数被
限制在声明它的模块的本地范围内使用。
总结:
	static全局变量与普通的全局变量有什么区别:static全局变量只初使化一次,防止在其他
	文件单元中被引用;
  static局部变量和普通局部变量有什么区别:static局部变量只被初始化一次,下一次依据
  上一次结果值;
  static函数与普通函数有什么区别:static函数在内存中只有一份,普通函数在每个被调用
  中维持一份拷贝. 
试题7.	如何引用一个已经定义过的全局变量?
  答案:
  可以用引用头文件的方式,也可以用extern关键字,如果用引用头文件方式来引用某个在头
  文件中声明的全局变理,假定你将那个变写错了,那么在编译期间会报错,如果你用extern
  方式引用时,假定你犯了同样的错误,那么在编译期间不会报错,而在连接期间报错。

试题8:在C++ 程序中调用被 C编译器编译后的函数,为什么要加 extern “C”? 
例子:
	#ifndef __INCvxWorksh
	#define __INCvxWorksh
	#ifdef __cplusplus
	extern "C" {
   
	#endif
	/*...*/
	#ifdef __cplusplus
	}
	#endif
	#endif /* __INCvxWorksh */ 
答案:
	由于c语言是没有重载函数的概念的,所以c编译器编译的程序里,所有函数只有函数名对应
	的入口。而由于c++语言有重载函数的概念,如果只有函数名对应的入口,则会出现混淆,所
	以c++编译器编译的程序,应该是函数名+参数类型列表对应到入口。
假设某个函数的原型为: void foo(int x, int y);
	该函数被C编译器编译后在库中的名字为_foo,而C++编译器则会产生像_foo_int_int之类
	的名字。
	C++提供了C连接交换指定符号extern“C”来解决名字匹配问题。

思考9:如何判断一段程序是由C 编译程序还是由C++编译程序编译的?
(1)如果是要你的代码在编译时发现编译器类型,就判断_cplusplus或_STDC_宏。 
#ifdef __cplusplus
       cout<<"c++";
#else
       cout<<"c";
#endif
 如果要判断已经编译的代码的编译类型,就用nm查一下输出函数符号是否和函数名相同。
(2)注意,因为mian函数是整个程序的入口,所以mian是不能有重载的,所以,如果一个程序只
有main函数,是无法确认是c还是c++编译器
编译的可以通过nm来查看函数名入口   
 如一个函数   
 int foo(int i, float j)
c编译的程序通过nm查看
foo           0x567xxxxxx    (地址)
c++编译程序,通过nm查看
foo(int, float)    0x567xxxxxx 
  另外,如果要在c++编译器里使用通过c编译的目标文件,必须通知c++编译器, extern  "c"  
  foo;

2.预处理命令(宏)

试题10:什么是预编译,何时需要预编译?
答: 
就是指程序执行前的一些预处理工作,主要指#表示的.)、总是使用不经常改动的大型代码体。
2)、程序由多个模块组成,所有模块都使用一组标准的包含文件和相同的编译选项。在这种情况
下,可以将所有包含文件预编译为一个预编译头。

试题11.说出以下 预处理器标识的目的意义

	   指令             用途
         #           
        #include   
        #define    
        #undef     
        #if         
        #ifdef      
        #ifndef     
        #elif       
        #endif              
		 #error      

答案:
	   指令             用途
         #           空指令,无任何效果
         #include    包含一个源代码文件
         #define     定义宏
         #undef      取消已定义的宏
         #if         如果给定条件为真,则编译下面代码
         #ifdef      如果宏已经定义,则编译下面代码
         #ifndef     如果宏没有定义,则编译下面代码
         #elif       如果前面的#if给定条件不为真,当前条件为真,则编译下面代码
         #endif      结束一个#if……#else条件编译块
         #error      停止编译并显示错误信息  编译时检测错误,
					使程序员更好的掌握代码-
试题12:头文件中的 ifndef/define/endif 干什么用?
	答:防止该头文件被重复引用。这样, 可以减少整个编译过程中打开这个文件的次数.
	等同于:
	#pragma once
试题13:#include “filename.h”和#include <filename.h>的区别?
答案:
	对于#include <filename.h>编译器从标准库开始搜索filename.h
    对于#include “filename.h”编译器从用户工作路径开始搜索filename.h
试题14: 用预处理指令#define 声明一个常数,用以表明1年中有多少秒(忽略闰年问题)
 答案:   #define SECONDS_PER_YEAR (60 * 60 * 24 * 365UL)
剖析:
	1) #define 语法的基本知识(例如:不能以分号结束,括号的使用,等等)
	2)懂得预处理器将为你计算常数表达式的值,因此,直接写出你是如何计算一年中有多少秒
	而不是计算出实际的值,是更清晰而没有代价的。
	3) 意识到这个表达式将使一个16位机的整型数溢出-因此要用到长整型符号L,告诉编译器这
	个常数是的长整型数。
	4) 再思考进一步用到UL(表示无符号长整型)
试题15. 写一个"标准"宏MIN ,这个宏输入两个参数并返回较小的一个。
	答案:
        #define MIN(A,B) ((A) <= (B) ? (A) : (B)) 

剖析:
	1) 标识#define在宏中应用的基本知识。这是很重要的。因为在 嵌入(inline)操作符 变
	为标准C的一部分之前,
	宏是方便产生嵌入代码的唯一方法,对于嵌入式系统来说,为了能达到要求的性能,嵌入式
	代码经常是必须的方法。
	2)三重条件操作符的知识。这个操作符存在C语言中的原因是它使得编译器能产生比
	if-then-else更优化的代码,了解这个用法是很重要的。
	3) 懂得在宏中小心地把参数用括号括起来
	注:谨慎地将宏定义中的“参数”和整个宏用用括弧括起来。所以,严格地讲,下述解答是错误
	的: 
	#define MIN(A,B) (A) <= (B) ? (A) : (B)
	#define MIN(A,B) (A <= B ? A : B ) 
	#define MIN(A,B) ((A) <= (B) ? (A) : (B)); //这个解答在宏定义的后面加“;”
	
试题16:	#define MIN(A,B) ((A)= (B) ? (A) : (B))MIN(*p++, b)的有什么后果? 
答案:
		((*p++)= (b) ? (*p++) : (*p++)) 
  这个表达式会产生副作用,指针p会作多次++自增操作。 

试题17:宏定义的多语句错误,分析以宏定义
		#define  D(a,b)  a+b;\
						a++;
		分析:
		应用时:if(XXX)
					D(a.b);
					else
		解决办法  用do{
    } while(0)
		#define  D(a,b)  do{a+b;\
								a++;}while(0)
		思考while(0)后没有分号
试题18:分析一下两个定义,哪种方法更好呢?(如果有的话)为什么?


		#define dPS struct s *
		typedef struct s * tPS;
分析:
	以上两种情况的意图都是要定义dPS 和 tPS 作为一个指向结构s指针。	
	答案是:typedef更好。思考下面的例子:

	dPS p1,p2;
	tPS p3,p4;

	第一个扩展为
	struct s * p1, p2;
	上面的代码定义p1为一个指向结构的指,p2为一个实际的结构,这也许不是你想要的。第二
	个例子正确地定义了p3 和p4 两个指针。
	假如定义函数指针:
		typedef   void (*fun)(void);
		#define   FUN(x)  void(*x)(void)

运算符·表达式·数据类型·优先级
试题19.分析以下代码,说出输出结果
#define swap(a,b) a=a+b;b=a-b;a=a-b; 
void main() 
{
    
	int x=5, y=10; 
	swap (x,y); 
	printf(%d %dn”,x,y); 
}
答案: 10, 5
试题20:分别给出BOOL,intfloat,指针变量 与“零值”比较的 if 语句(假设变量名为
var) 
分析:
 BOOL型变量:if(!var)
 int型变量: if(var==0)
 float型变量:
 const float EPSINON = 0.00001;
 if ((x >= - EPSINON) && (x <= EPSINON)
 指针变量:  if(var==NULL) 
  
分析: 
  考查对0值判断的“内功”,BOOL型变量的0判断完全可以写成if(var==0),而int型变量也可
  以写成if(!var),
	指针变量的判断也可以写成if(!var),上述写法虽然程序都能正确运行,但是未能清晰地表
	达程序的意思。 
  一般的,如果想让if判断一个变量的“真”、“假”,应直接使用if(var)if(!var),表明其
  为“逻辑”判断;如果用if判断一个数值型变量(shortintlong),
	应该用if(var==0),表明是与0进行“数值”上的比较;而判断指针则适宜用
	if(var==NULL),这是一种很好的编程习惯。 
  浮点型变量并不精确,所以不可将float变量用“==”或“!=”与数字比较,应该设法转化成“>=”
  或“<=”形式。如果写成if (x == 0.0),则判为错 
试题21: 
	嵌入式系统总是要用户对变量或寄存器进行位操作。给定一个整型变量a,写两段代码,第一个设置a的bit 3,第二个清除a 的bit 3。在以上两个操作中,要保持其它位不变。

分析:
	#define BIT3 (0x1 << 3)
	static int a;
	void set_bit3(void) 
	{
   
		a |= BIT3;
	}
	void clear_bit3(void) 
	{
   
		a &= ~BIT3;
	}	
补充22:了解可以位操作的另一个知识位域
	有些信息在存储时,并不需要占用一个完整的字节, 而只需占几个或一个二进制位。
	例如在存放一个开关量时,只有01 两种状态, 用一位二进位即可。
	为了节省存储空间,并使处理简便,C语言又提供了一种数据结构,称为“位域”或“位段”。
	所谓“位域”是把一个字节中的二进位划分为几个不同的区域, 并说明每个区域的位数。
	每个域有一个域名,允许在程序中按域名进行操作。 这样就可以把几个不同的对象用一个字
	节的二进制位域来表示。
	一、位域的定义和位域变量的说明位域定义与结构定义相仿,其形式为: 
	struct 位域结构名 {
    位域列表 }; 其中位域列表的形式为: 类型说明符 位域名:位域长度 
  例如: 
  struct bs 
  {
    
   int a:8; 
   int b:2; 
   int c:
  • 5
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值