求职:嵌入式程C语言笔试面试题

上传人:回**** 文档编号:148104172 上传时间:2022-09-04 格式:DOCX 页数:22 大小:39.24KB
收藏 版权申诉 举报 下载
求职:嵌入式程C语言笔试面试题_第1页
第1页 / 共22页
求职:嵌入式程C语言笔试面试题_第2页
第2页 / 共22页
求职:嵌入式程C语言笔试面试题_第3页
第3页 / 共22页
资源描述:

《求职:嵌入式程C语言笔试面试题》由会员分享,可在线阅读,更多相关《求职:嵌入式程C语言笔试面试题(22页珍藏版)》请在装配图网上搜索。

1、2022求职:嵌入式程C语言笔试面试题求职:嵌入式程C语言笔试(面试)题C语言测试是聘请嵌入式系统程序员以及电子限制类专业求职过程中必需面临的。从被面试者的角度来讲,你能了解很多关于出题者或监考者的状况。这个测试只是出题者为显示其对ANSI标准细微环节的学问而不是技术技巧而设计吗?这是个愚蠢的问题吗?如要你答出某个字符的ASCII值。这些问题着重考察你的系统调用和内存安排策略方面的实力吗?这标记着出题者或许花时间在微机上而不是在嵌入式系统上。假如上述任何问题的答案是是的话,那么我知道我得仔细考虑我是否应当去做这份工作。从面试者的角度来讲,一个测试或许能从多方面揭示应试者的素养:最基本的,你能了

2、解应试者C语言的水平。不管怎么样,看一下这人如何回答他不会的问题也是满好玩。应试者是以好的直觉做出明智的选择,还是只是瞎蒙呢?当应试者在某个问题上卡住时是找借口呢,还是表现出对问题的真正的新奇心,把这看成学习的机会呢?我发觉这些信息与他们的测试成果一样有用。下面这些令人头痛的考题能给正在找工作的人一点帮助。其中有些题很难,但它们应当都能给你一点启迪。这个测试适于不同水平的应试者,大多数初级水平的应试者的成果会很差,阅历丰富的程序员应当有很好的成果。为了让你能自己确定某些问题的偏好,每个问题没有安排分数,假如选择这些考题为你所用,请自行按你的意思安排分数。预处理器(preprocessor)1

3、. 用预处理指令#define 声明一个常数,用以表明1年中有多少秒(忽视闰年问题)#define SECONDS_pER_YEAR (60 * 60 * 24 * 365)UL我在这想看到几件事情:1) #define 语法的基本学问(例如:不能以分号结束,括号的运用,等等)2)懂得预处理器将为你计算常数表达式的值,因此,干脆写出你是如何计算一年中有多少秒而不是计算出实际的值,是更清楚而没有代价的。3) 意识到这个表达式将使一个16位机的整型数溢出-因此要用到长整型符号L,告知编译器这个常数是的长整型数。4) 假如你在你的表达式中用到UL(表示无符号长整型),那么你有了一个好的起点。记住,第

4、一印象很重要。2 . 写一个标准宏MIN ,这个宏输入两个参数并返回较小的一个。#define MIN(A,B) (A) <= (B) ? (A) : (B) 这个测试是为下面的目的而设的:1) 标识#define在宏中应用的基本学问。这是很重要的。因为在 嵌入(inline)操作符 变为标准C的一部分之前,宏是便利产生嵌入代码的唯一方法,对于嵌入式系统来说,为了能达到要求的性能,嵌入代码常常是必需的方法。2)三重条件操作符的学问。这个操作符存在C语言中的缘由是它使得编译器能产生比if-then-else更优化的代码,了解这个用法是很重要的。3) 懂得在宏中当心地把参数用括号括起来4)

5、我也用这个问题起先探讨宏的副作用,例如:当你写下面的代码时会发生什么事?least = MIN(*p+, b);3. 预处理器标识#error的目的是什么?假如你不知道答案,请看参考文献。这问题对区分一个正常的店员和一个书呆子是很有用的。只有书呆子才会读C语言课本的附录去找出象这种问题的答案。当然假如你不是在找一个书呆子,那么应试者最好希望自己不要知道答案。死循环(Infinite loops)4. 嵌入式系统中常常要用到无限循环,你怎么样用C编写死循环呢?这个问题用几个解决方案。我首选的方案是:while(1)一些程序员更喜爱如下方案:for(;)这个实现方式让我犯难,因为这个语法没有准确表

6、达究竟怎么回事。假如一个应试者给出这个作为方案,我将用这个作为一个机会去探究他们这样做的基本原理。假如他们的基本答案是:我被教着这样做,但从没有想到过为什么。这会给我留下一个坏印象。第三个方案是用 gotoLoop:.goto Loop;应试者如给出上面的方案,这说明或者他是一个汇编语言程序员(这或许是好事)或者他是一个想进入新领域的BASIC/FORTRAN程序员。数据声明(Data declarations)5. 用变量a给出下面的定义a) 一个整型数(An integer)b)一个指向整型数的指针( A pointer to an integer)c)一个指向指针的的指针,它指向的指针是

7、指向一个整型数( A pointer to a pointer to an intege)rd)一个有10个整型数的数组( An array of 10 integers)e) 一个有10个指针的数组,该指针是指向一个整型数的。(An array of 10 pointers tointegers)f) 一个指向有10个整型数数组的指针( A pointer to an array of 10 integers)g) 一个指向函数的指针,该函数有一个整型参数并返回一个整型数(A pointer to a functionthat takes an integer as an argument

8、and returns an integer)h) 一个有10个指针的数组,该指针指向一个函数,该函数有一个整型参数并返回一个整型数( An array of ten pointers to functions that take an integer argument and return an integer )答案是:a) int a; / An integerb) int *a; / A pointer to an integerc) int *a; / A pointer to a pointer to an integerd) int a10; / An array of 10 i

9、ntegerse) int *a10; / An array of 10 pointers to integersf) int (*a)10; / A pointer to an array of 10 integersg) int (*a)(int); / A pointer to a function a thattakes an integer argument and returns an integerh) int (*a10)(int); / An array of 10 pointers tofunctions that take an integer argument and

10、return aninteger人们常常声称这里有几个问题是那种要翻一下书才能回答的问题,我同意这种说法。当我写这篇文章时,为了确定语法的正确性,我的确查了一下书。但是当我被面试的时候,我期望被问到这个问题(或者相近的问题)。因为在被面试的这段时间里,我确定我知道这个问题的答案。应试者假如不知道全部的答案(或至少大部分答案),那么也就没有为这次面试做打算,假如该面试者没有为这次面试做打算,那么他又能为什么出打算呢?Static6. 关键字static的作用是什么?这个简洁的问题很少有人能回答完全。在C语言中,关键字static有三个明显的作用:1)在函数体,一个被声明为静态的变量在这一函数被调

11、用过程中维持其值不变。2)在模块内(但在函数体外),一个被声明为静态的变量可以被模块内所用函数访问,但不能被模块外其它函数访问。它是一个本地的全局变量。3) 在模块内,一个被声明为静态的函数只可被这一模块内的其它函数调用。那就是,这个函数被限制在声明它的模块的本地范围内运用。大多数应试者能正确回答第一部分,一部分能正确回答其次部分,同是很少的人能懂得第三部分。这是一个应试者的严峻的缺点,因为他明显不懂得本地化数据和代码范围的好处和重要性。Const7关键字const有什么含意?我只要一听到被面试者说:const意味着常数,我就知道我正在和一个业余者打交道。去年DanSaks 已经在他的文章里完

12、全概括了const的全部用法,因此ESp(译者:Embedded Systems programming)的每一位读者应当特别熟识const能做什么和不能做什么.假如你从没有读到那篇文章,只要能说出const意味着只读就可以了。尽管这个答案不是完全的答案,但我接受它作为一个正确的答案。(假如你想知道更具体的答案,细致读一下Saks的文章吧。)假如应试者能正确回答这个问题,我将问他一个附加的问题:下面的声明都是什么意思?const int a;int const a;const int *a;int * const a;int const * a const;/*/前两个的作用是一样,a是一个常

13、整型数。第三个意味着a是一个指向常整型数的指针(也就是,整型数是不行修改的,但指针可以)。第四个意思a是一个指向整型数的常指针(也就是说,指针指向的整型数是可以修改的,但指针是不行修改的)。最终一个意味着a是一个指向常整型数的常指针(也就是说,指针指向的整型数是不行修改的,同时指针也是不行修改的)。假如应试者能正确回答这些问题,那么他就给我留下了一个好印象。顺带提一句,或许你可能会问,即使不用关键字 ,也还是能很简单写出功能正确的程序,那么我为什么还要如此看重关键字const呢?我也如下的几下理由:1) 关键字const的作用是为给读你代码的人传达特别有用的信息,事实上,声明一个参数为常量是为

14、了告知了用户这个参数的应用目的。假如你曾花许多时间清理其它人留下的垃圾,你就会很快学会感谢这点多余的信息。(当然,懂得用const的程序员很少会留下的垃圾让别人来清理的。)2) 通过给优化器一些附加的信息,运用关键字const或许能产生更紧凑的代码。3) 合理地运用关键字const可以使编译器很自然地爱护那些不希望被变更的参数,防止其被无意的代码修改。简而言之,这样可以削减bug的出现。Volatile8. 关键字volatile有什么含意?并给出三个不同的例子。一个定义为volatile的变量是说这变量可能会被意想不到地变更,这样,编译器就不会去假设这个变量的值了。精确地说就是,优化器在用到

15、这个变量时必需每次都当心地重新读取这个变量的值,而不是运用保存在寄存器里的备份。下面是volatile变量的几个例子:1) 并行设备的硬件寄存器(如:状态寄存器)2) 一个中断服务子程序中会访问到的非自动变量(Non-automatic variables)3) 多线程应用中被几个任务共享的变量回答不出这个问题的人是不会被雇佣的。我认为这是区分C程序员和嵌入式系统程序员的最基本的问题。搞嵌入式的家伙们常常同硬件、中断、RTOS等等打交道,全部这些都要求用到volatile变量。不懂得volatile的内容将会带来灾难。 假设被面试者正确地回答了这是问题(嗯,怀疑是否会是这样),我将略微深究一下

16、,看一下这家伙是不是直正懂得volatile完全的重要性。1)一个参数既可以是const还可以是volatile吗?说明为什么。2); 一个指针可以是volatile 吗?说明为什么。3); 下面的函数有什么错误:int square(volatile int *ptr)return *ptr * *ptr;下面是答案:1)是的。一个例子是只读的状态寄存器。它是volatile因为它可能被意想不到地变更。它是const因为程序不应当试图去修改它。2); 是的。尽管这并不很常见。一个例子是当一个中服务子程序修该一个指向一个buffer的指针时。3) 这段代码有点变态。这段代码的目的是用来返指针*

17、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;位操作(Bit manipulation)9. 嵌入式系统总是要用户对变量或寄存器进行位操作。给定一个整型变量a,写两段代码,第一

18、个设置a的bit 3,其次个清除a 的bit 3。在以上两个操作中,要保持其它位不变。 对这个问题有三种基本的反应1)不知道如何下手。该被面者从没做过任何嵌入式系统的工作。2) 用bit fields。Bit fields是被扔到C语言死角的东西,它保证你的代码在不同编译器之间是不行移植的,同时也保证了的你的代码是不行重用的。我最近不幸看到 Infineon为其较困难的通信芯片写的驱动程序,它用到了bit fields因此完全对我无用,因为我的编译器用其它的方式来实现bit fields的。从道德讲:恒久不要让一个非嵌入式的家伙粘实际硬件的边。3) 用 #defines 和 bit masks

19、 操作。这是一个有极高可移植性的方法,是应当被用到的方法。最佳的解决方案如下:#define BIT3 (0x1 << 3)static int a;void set_bit3(void)a |= BIT3;void clear_bit3(void)a &= BIT3;一些人喜爱为设置和清除值而定义一个掩码同时定义一些说明常数,这也是可以接受的。我希望看到几个要点:说明常数、|=和&=操作。访问固定的内存位置(Accessing fixed memory locations)10. 嵌入式系统常常具有要求程序员去访问某特定的内存位置的特点。在某工程中,要求设置一肯定

20、地址为0x67a9的整型变量的值为0xaa66。编译器是一个纯粹的ANSI编译器。写代码去完成这一任务。这一问题测试你是否知道为了访问一肯定地址把一个整型数强制转换(typecast)为一指针是合法的。这一问题的实现方式随着个人风格不同而不同。典型的类似代码如下:int *ptr;ptr = (int *)0x67a9;*ptr = 0xaa55;A more obscure approach is: ( 一个较晦涩的方法是):*(int * const)(0x67a9) = 0xaa55;即使你的品尝更接近其次种方案,但我建议你在面试时运用第一种方案。中断(Interrupts)11. 中断

21、是嵌入式系统中重要的组成部分,这导致了许多编译开发商供应一种扩展让标准C支持中断。具代表事实是,产生了一个新的关键字 _interrupt。下面的代码就运用了_interrupt关键字去定义了一个中断服务子程序(ISR),请评论一下这段代码的。_interrupt double compute_area (double radius)double area = pI * radius * radius;printf(nArea = %f, area);return area;这个函数有太多的错误了,以至让人不知从何说起了:1)ISR 不能返回一个值。假如你不懂这个,那么你不会被雇用的。2) I

22、SR 不能传递参数。假如你没有看到这一点,你被雇用的机会等同第一项。3) 在很多的处理器/编译器中,浮点一般都是不行重入的。有些处理器/编译器须要让额处的寄存器入栈,有些处理器/编译器就是不允许在ISR中做浮点运算。此外,ISR应当是短而有效率的,在ISR中做浮点运算是不明智的。4) 与第三点一脉相承,printf()常常有重入和性能上的问题。假如你丢掉了第三和第四点,我不会太犯难你的。不用说,假如你能得到后两点,那么你的被雇用前景越来越光明白。代码例子(Code examples)12 . 下面的代码输出是什么,为什么?void foo(void)unsigned int a = 6;int

23、 b = -20;(a+b > 6) ? puts(> 6) : puts(<= 6);这个问题测试你是否懂得C语言中的整数自动转换原则,我发觉有些开发者懂得极少这些东西。不管如何,这无符号整型问题的答案是输出是 >6。缘由是当表达式中存在有符号类型和无符号类型时全部的操作数都自动转换为无符号类型。因此-20变成了一个特别大的正整数,所以该表达式计算出的结果大于 6。这一点对于应当频繁用到无符号数据类型的嵌入式系统来说是丰常重要的。假如你答错了这个问题,你也就到了得不到这份工作的边缘。13. 评价下面的代码片断:unsigned int zero = 0;unsigne

24、d int compzero = 0xFFFF;/*1s complement of zero */对于一个int型不是16位的处理器为说,上面的代码是不正确的。应编写如下:unsigned int compzero = 0;这一问题真正能揭露出应试者是否懂得处理器字长的重要性。在我的阅历里,好的嵌入式程序员特别精确地明白硬件的细微环节和它的局限,然而pC机程序往往把硬件作为一个无法避开的苦恼。到了这个阶段,应试者或者完全没精打采了或者信念满满志在必得。假如明显应试者不是很好,那么这个测试就在这里结束了。但假如明显应试者做得不错,那么我就扔出下面的追加问题,这些问题是比较难的,我想仅仅特别优秀

25、的应试者能做得不错。提出这些问题,我希望更多看到应试者应付问题的方法,而不是答案。不管如何,你就当是这个消遣吧.动态内存安排(Dynamic memory allocation)14. 尽管不像非嵌入式计算机那么常见,嵌入式系统还是有从堆(heap)中动态安排内存的过程的。那么嵌入式系统中,动态安排内存可能发生的问题是什么?这里,我期望应试者能提到内存碎片,碎片收集的问题,变量的持行时间等等。这个主题已经在ESp杂志中被广泛地探讨过了(主要是 p.J. plauger, 他的说明远远超过我这里能提到的任何说明),全部回过头看一下这些杂志吧!让应试者进入一种虚假的平安感觉后,我拿出这么一个小节目

26、:下面的代码片段的输出是什么,为什么?char *ptr;if (ptr = (char *)malloc(0) = NULL)puts(Got a null pointer);elseputs(Got a valid pointer);这是一个好玩的问题。最近在我的一个同事不经意把0值传给了函数malloc,得到了一个合法的指针之后,我才想到这个问题。这就是上面的代码,该代码的输出是Got a valid pointer。我用这个来起先探讨这样的一问题,看看被面试者是否想到库例程这样做是正确。得到正确的答案当然重要,但解决问题的方法和你做确定的基本原理更重要些。Typedef15 Typed

27、ef 在C语言中频繁用以声明一个已经存在的数据类型的同义字。也可以用预处理器做类似的事。例如,思索一下下面的例子:#define dpS struct s *typedef struct s * tpS;以上两种状况的意图都是要定义dpS 和 tpS 作为一个指向结构s指针。哪种方法更好呢?(假如有的话)为什么?这是一个特别微妙的问题,任何人答对这个问题(正值的缘由)是应当被恭喜的。答案是:typedef更好。思索下面的例子:dpS p1,p2;tpS p3,p4;第一个扩展为struct s * p1, p2;上面的代码定义p1为一个指向结构的指,p2为一个实际的结构,这或许不是你想要的。其

28、次个例子正确地定义了p3 和p4 两个指针。晦涩的语法16 . C语言同意一些令人震惊的结构,下面的结构是合法的吗,假如是它做些什么?int a = 5, b = 7, c;c = a+b;这个问题将做为这个测验的一个开心的结尾。不管你相不信任,上面的例子是完全合乎语法的。问题是编译器如何处理它?水平不高的编译作者事实上会争辩这个问题,依据最处理原则,编译器应当能处理尽可能全部合法的用法。因此,上面的代码被处理成:c = a+ + b;因此, 这段代码持行后a = 6, b = 7, c = 12。假如你知道答案,或猜出正确答案,做得好。假如你不知道答案,我也不把这个当作问题。我发觉这个问题的

29、最大好处是这是一个关于代码编写风格,代码的可读性,代码的可修改性的好的话题。 好了,店员们,你现在已经做完全部的测试了。这就是我出的C语言测试题,我怀着开心的心情写完它,希望你以同样的心情读完它。假如是认为这是一个好的测试,那么尽量都用到你的找工作的过程中去吧。 本文来源:网络收集与整理,如有侵权,请联系作者删除,谢谢!第22页 共22页第 22 页 共 22 页第 22 页 共 22 页第 22 页 共 22 页第 22 页 共 22 页第 22 页 共 22 页第 22 页 共 22 页第 22 页 共 22 页第 22 页 共 22 页第 22 页 共 22 页第 22 页 共 22 页

展开阅读全文
温馨提示:
1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
2: 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
3.本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
关于我们 - 网站声明 - 网站地图 - 资源地图 - 友情链接 - 网站客服 - 联系我们

copyright@ 2023-2025  zhuangpeitu.com 装配图网版权所有   联系电话:18123376007

备案号:ICP2024067431-1 川公网安备51140202000466号


本站为文档C2C交易模式,即用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知装配图网,我们立即给予删除!