志在指尖
用双手敲打未来

c语言基础知识(题库)

c语言基础知识

c言语入门根底知识点有:1、c言语的命名;2、变量及赋值;3、根本数据类型;4、格式化输出句子;5、强制类型转化;6、常量;7、主动类型转化。
【推荐课程:C言语教程】
(1)c言语的命名
编程时给变量或许函数起的姓名便是标识符。C言语的标识符是不行以随意起姓名的,有必要恪守一定的规矩。C言语规矩,标识符可所以字母(A~Z,a~z)、数字(0~9)、下划线_组成的字符串,而且第一个字符有必要是字母或下划线。在运用标识符时还有留意以下几点:
(1)标识符的长度最好不要超越8位,由于在某些版别的C中规矩标识符前8位有效,当两个标识符前8位相同时,则被认为是同一个标识符。
(2)标识符是严厉区分大小写的。例如Imooc和imooc是两个不同的标识符。
(3)标识符最好选择有意义的英文单词组成做到”见名知意”,不要运用中文。
(4)标识符不能是C言语的关键字。想了解更多C言语关键字的知识,请查阅WIKI。
例如:
(2)变量及赋值
变量便是能够改动的量,而每个变量都会有一个姓名(标识符)。变量占据内存中一定的存储单元。运用变量之前有必要先界说变量,要区分变量名和变量值是两个不同的概念。就好比:住在房间里的客人与房间号是两个不同的概念。
变量界说的一般方式为:数据类型变量名;
多个类型相同的变量:数据类型变量名,变量名,变量名…;
变量名和标识符的命名规范完全相同。%d在后面会有详细讲解
留意:在界说中不允许连续赋值,如inta=b=c=5;是不合法的。c
变量的赋值分为两种方法:
1.先声明再赋值
2.声明的同时赋值
(3)根本数据类型
C言语中的数据也是有类型的,C言语中,数据类型可分为:根本数据类型,结构数据类型,指针类型,空类型四大类。如图所示:
例:
(4)格式化输出句子
格式化输出句子,也能够说是占位输出,是将各种类型的数据依照格式化后的类型及指定的位置从计算机上显现。这样做的优点,是有利于计算机能够精确的将咱们所要类型数据给咱们。
其格式为:
1
printf(“输出格式符”,输出项);
C言语中的常用格式化符:
(5)不行改动的常量
在程序履行过程中,值不发生改动的量称为常量。C言语的常量能够分为直接常量和符号常量。
直接常量也称为字面量,是能够直接拿来运用,无需说明的量,比方:
整型常量:13、0、-13;
实型常量:13.33、-24.4;
字符常量:‘a’、‘M’
字符串常量:”Ilovephp!”
在C言语中,能够用一个标识符来表明一个常量,称之为符号常量。符号常量在运用之前有必要先界说,其一般方式为:
1
#define标识符常量值
符号常量的标明符一般习惯运用大写字母,变量的标明符一般习惯运用小写字母,加以区分。下面是一个运用符号常量的小比如:
(6)主动类型转化
数据类型存在主动转化的情况,主动转化发生在不同数据类型运算时,在编译的时候主动完结。主动转化遵从的规矩就好比小盒子能够放进大盒子里边一样,下图表明了类型主动转化的规矩。
(7)强制类型转化
强制类型转化是经过界说类型转化运算来实现的。其一般方式为:
1
(数据类型)(表达式)
其作用是把表达式的运算成果强制转化成类型说明符所表明的类型,例如:
留意:double类型的数据的格式符%f或%lf
输入成果:
在运用强制转化时应留意以下问题:
1、数据类型和表达式都有必要加括号,如把(int)(x/2+y)写成(int)x/2+y则成了把x转化成int型之后再除2再与y相加了。
2、转化后不会改动原数据的类型及变量值,只在本次运算中临时性转化。
3、强制转化后的运算成果不遵从四舍五入原则。
(8)运算符
那么C言语中又有哪些运算符呢?
如下所示:
算术运算符
赋值运算符
联系运算符
逻辑运算符
三目运算符
算术运算法
C言语根本算术运算符如下表:
赋值运算符
C言语中赋值运算符分为简略赋值运算符和复合赋值运算符,之前咱们现已触摸过简略赋值运算符“=”号了,下面讲一下复合赋值运算符:
复合赋值运算符便是在简略赋值符“=”之前加上其它运算符构成,例如+=、-=、*=、/=、%=。
看一个小比如:
1
2
inta=3;
a+=5;
分析:界说整型变量a并赋值为3,a+=5;这个算式就等价于a=a+5;将变量a和5相加之后再赋值给a
留意:复合运算符中运算符和等号之间是不存在空格的。
联系运算符
下面是C言语中的联系运算符:
联系表达式的值是“真”和“假”,在C程序用整数1和0表明。
逻辑运算符
下面咱们看一下C言语中的逻辑运算符:
三木运算符
C言语中的三目运算符:“?:”,其格式为:
1
表达式1?表达式2:表达式3;
运算符优先级比较

c语言基础知识题库

1、用预处理指令#define声明一个常数,用以标明1年中有多少秒(疏忽闰年问题)
#defineSECONDS_PER_YEAR(60*60*24*365)UL
我在这想看到几件事情:
1).#define语法的根本知识(例如:不能以分号完毕,括号的运用,等等)
2).懂得预处理器将为你计算常数表达式的值,因而,直接写出你是怎么计算一年中有多少秒而不是计算出实践的值,是更明晰而没有代价的。
3).意识到这个表达式将使一个16位机的整型数溢出-因而要用到长整型符号L,告诉编译器这个常数是的长整型数。
4).假如你在你的表达式中用到UL(表示无符号长整型),那么你有了一个好的起点。记住,榜首形象很重要。
2、写一个“规范”宏MIN,这个宏输入两个参数并返回较小的一个。
#defineMIN(A,B)((A)<=(B)?(A):(B)))
这个检验是为下面的意图而设的:
1).标识#define在宏中应用的根本知识。这是很重要的,由于直到嵌入(inline)操作符变为规范C的一部分,宏是方便产生嵌入代码的唯一办法,关于嵌入式体系来说,为了能达到要求的功能,嵌入代码常常是有必要的办法。
2).三重条件操作符的知识。这个操作符存在C言语中的原因是它使得编译器能产生比if-then-else更优化的代码,了解这个用法是很重要的。
3).懂得在宏中小心地把参数用括号括起来。
4).我也用这个问题开端评论宏的副作用,例如:当你写下面的代码时会产生什么事?
Least=MIN(*p++,b);
3、预处理器标识#error的意图是什么?
这问题对区别一个正常的伙计和一个书呆子是很有用的。只需书呆子才会读C言语讲义的附录去找出象这种问题的答案。当然假如你不是在找一个书呆子,那么应试者最好希望自己不要知道答案。
正确答案:
编译程序时,只需遇到#error就会跳出一个编译过错。既然是编译过错,要它干嘛呢?其意图便是确保程序是按照你所设想的那样进行编译的。c1
4、嵌入式体系中常常要用到无限循环,你怎么样用C编写死循环呢?
这个问题用几个解决计划。我首选的计划是:
while(1){}
一些程序员更喜欢如下计划:
for(;;){}
这个完成办法让我尴尬,由于这个语法没有确切表达究竟怎么回事。假如一个应试者给出这个作为计划,我将用这个作为一个机会去探究他们这样做的根本原理。假如他们的根本答案是:“我被教着这样做,但从没有想到过为什么。”这会给我留下一个坏形象。
第三个计划是用goto
Loop:…gotoLoop;
应试者如给出上面的计划,这说明或许他是一个汇编言语程序员(这或许是功德)或许他是一个想进入新领域的BASIC/FORTRAN程序员。
5、用变量a给出下面的界说
a)一个整型数(Aninteger)
b)一个指向整型数的指针(Apointertoaninteger)
c)一个指向指针的的指针,它指向的指针是指向一个整型数(Apointertoapointertoaninteger)
d)一个有10个整型数的数组(Anarrayof10integers)
e)一个有10个指针的数组,该指针是指向一个整型数的(Anarrayof10pointerstointegers)
f)一个指向有10个整型数数组的指针(Apointertoanarrayof10integers)
g)一个指向函数的指针,该函数有一个整型参数并返回一个整型数(Apointertoafunctionthattakesanintegerasanargumentandreturnsaninteger)
h)一个有10个指针的数组,该指针指向一个函数,该函数有一个整型参数并返回一个整型数(Anarrayoftenpointerstofunctionsthattakeanintegerargumentandreturnaninteger)
答案是:
a)inta;//Aninteger
b)int*a;//Apointertoaninteger
c)int**a;//Apointertoapointertoaninteger
d)inta[10];//Anarrayof10integers
e)int*a[10];//Anarrayof10pointerstointegers
f)int(*a)[10];//Apointertoanarrayof10integers
g)int(*a)(int);//Apointertoafunctionathattakesanintegerargumentandreturnsaninteger
h)int(*a[10])(int);//Anarrayof10pointerstofunctionsthattakeanintegerargumentandreturnaninteger
人们常常宣称这儿有几个问题是那种要翻一下书才干答复的问题,我赞同这种说法。当我写这篇文章时,为了确认语法的正确性,我确实查了一下书。
但是当我被面试的时候,我希望被问到这个问题(或许附近的问题)。由于在被面试的这段时刻里,我确认我知道这个问题的答案。应试者假如不知道一切的答案(或至少大部分答案),那么也就没有为这次面试做准备,假如该面试者没有为这次面试做准备,那么他又能为什么出准备呢?
6、关键字static的作用是什么?
这个简略的问题很少有人能答复彻底。
在C言语中,关键字static有三个明显的作用:
1).在函数体,一个被声明为静态的变量在这一函数被调用过程中维持其值不变。
2).在模块内(但在函数体外),一个被声明为静态的变量能够被模块内所用函数拜访,但不能被模块外其它函数拜访。它是一个本地的全局变量。
3).在模块内,一个被声明为静态的函数只可被这一模块内的其它函数调用。那便是,这个函数被约束在声明它的模块的本地范围内运用。
大多数应试者能正确答复榜首部分,一部分能正确答复第二部分,同是很少的人能懂得第三部分。这是一个应试者的严峻的缺陷,由于他明显不懂得本地化数据和代码范围的好处和重要性。
7、关键字const是什么含义?
我只需一听到被面试者说:“const意味着常数”,我就知道我正在和一个业余者打交道。去年DanSaks现已在他的文章里彻底概括了const的一切用法,因而ESP(译者:EmbeddedSystemsProgramming)的每一位读者应该十分熟悉const能做什么和不能做什么。
假如你从没有读到那篇文章,只需能说出const意味着“只读”就能够了。虽然这个答案不是彻底的答案,但我承受它作为一个正确的答案。(假如你想知道更详细的答案,仔细读一下Saks的文章吧。)假如应试者能正确答复这个问题,我将问他一个附加的问题:下面的声明都是什么意思?
Constinta;intconsta;constint*a;int*consta;intconst*aconst;
前两个的作用是一样,a是一个常整型数。第三个意味着a是一个指向常整型数的指针(也便是,整型数是不行修正的,但指针能够)。
第四个意思a是一个指向整型数的常指针(也便是说,指针指向的整型数是能够修正的,但指针是不行修正的)。最后一个意味着a是一个指向常整型数的常指针(也便是说,指针指向的整型数是不行修正的,同时指针也是不行修正的)。
假如应试者能正确答复这些问题,那么他就给我留下了一个好形象。顺带提一句,或许你或许会问,即便不必关键字const,也仍是能很容易写出功能正确的程序,那么我为什么还要如此看重关键字const呢?
我也如下的几下理由:
1).关键字const的作用是为给读你代码的人传达十分有用的信息,实践上,声明一个参数为常量是为了告诉了用户这个参数的应用意图。假如你曾花许多时刻整理其它人留下的废物,你就会很快学会感谢这点剩余的信息。(当然,懂得用const的程序员很少会留下的废物让他人来整理的。)
2).通过给优化器一些附加的信息,运用关键字const或许能产生更紧凑的代码。
3).合理地运用关键字const能够使编译器很自然地维护那些不希望被改动的参数,防止其被无意的代码修正。简而言之,这样能够减少bug的出现。
8、关键字volatile有什么含义并给出三个不同的比如。
一个界说为volatile的变量是说这变量或许会被意想不到地改动,这样,编译器就不会去假设这个变量的值了。精确地说便是,优化器在用到这个变量时有必要每次都小心地从头读取这个变量的值,而不是运用保存在寄存器里的备份。
下面是volatile变量的几个比如:
1).并行设备的硬件寄存器(如:状态寄存器)
2).一个中止服务子程序中会拜访到的非自动变量(Non-automaticvariables)
3).多线程应用中被几个使命共享的变量
答复不出这个问题的人是不会被雇佣的。我以为这是区别C程序员和嵌入式体系程序员的最根本的问题。嵌入式体系程序员常常同硬件、中止、RTOS等等打交道,所用这些都要求volatile变量。不懂得volatile内容将会带来灾难。
假设被面试者正确地答复了这是问题(嗯,置疑这否会是这样),我将略微深究一下,看一下这家伙是不是直正懂得volatile彻底的重要性。
1).一个参数既能够是const还能够是volatile吗?解说为什么。
2).一个指针能够是volatile吗?解说为什么。
3).下面的函数有什么过错:
intsquare(volatileint*ptr){returnptr*ptr;}
下面是答案:
1).是的。一个比如是只读的状态寄存器。它是volatile由于它或许被意想不到地改动。它是const由于程序不应该试图去修正它。
2).是的。虽然这并不很常见。一个比如是当一个中服务子程序修该一个指向一个buffer的指针时。
3).这段代码的有个恶作剧。这段代码的意图是用来返指针*ptr指向值的平方,但是,由于*ptr指向一个volatile型参数,编译器将产生相似下面的代码:
intsquare(volatileint*ptr){inta,b;a=*ptr;b=*ptr;returna*b;}
由于*ptr的值或许被意想不到地该变,因而a和b或许是不同的。成果,这段代码或许返不是你所希望的平方值!正确的代码如下:
longsquare(volatileint*ptr){inta;a=*ptr;returna*a;}
9、嵌入式体系总是要用户对变量或寄存器进行位操作。给定一个整型变量a,写两段代码,榜首个设置a的bit3,第二个清除a的bit3。在以上两个操作中,要坚持其它位不变。
对这个问题有三种根本的反响:
1).不知道怎么下手。该被面者从没做过任何嵌入式体系的作业。
2).用bitfields。Bitfields是被扔到C言语死角的东西,它确保你的代码在不同编译器之间是不行移植的,同时也确保了的你的代码是不行重用的。我最近不幸看到Infineon为其较杂乱的通讯芯片写的驱动程序,它用到了bitfields因而彻底对我无用,由于我的编译器用其它的办法来完成bitfields的。从品德讲:永久不要让一个非嵌入式的家伙粘实践硬件的边。
3).用#defines和bitmasks操作。这是一个有极高可移植性的办法,是应该被用到的办法。最佳的解决计划如下:
#defineBIT3(0x1<<3)staticinta;voidset_bit3(void){a|=BIT3;}voidclear_bit3(void){a&=~BIT3;}
一些人喜欢为设置和清除值而界说一个掩码同时界说一些说明常数,这也是能够承受的。我希望看到几个关键:说明常数、|=和&=~操作。
10、嵌入式体系常常具有要求程序员去拜访某特定的内存方位的特点。在某工程中,要求设置一绝对地址为0x67a9的整型变量的值为0xaa66。编译器是一个纯粹的ANSI编译器。写代码去完结这一使命。
这一问题检验你是否知道为了拜访一绝对地址把一个整型数强制转化(typecast)为一指针是合法的。这一问题的完成办法跟着个人风格不同而不同。典型的相似代码如下:
int*ptr;ptr=(int*)0x67a9;*ptr=0xaa55;
一个较晦涩的办法是:
*(int*const)(0x67a9)=0xaa55;
即便你的品味更接近第二种计划,但我主张你在面试时运用榜首种计划。
11、中止是嵌入式体系中重要的组成部分,这导致了许多编译开发商供给一种扩展—让规范C支持中止。具代表事实是,产生了一个新的关键字__interrupt。下面的代码就运用了__interrupt关键字去界说了一个中止服务子程序(ISR),请评论一下这段代码的。
__interruptdoublecompute_area(doubleradius){doublearea=PI*radius*radius;printf(”Area=%f”,area);returnarea;}这个函数有太多的过错了,以致让人不知从何说起了:1).ISR不能返回一个值。假如你不懂这个,那么你不会被招聘的。2).ISR不能传递参数。假如你没有看到这一点,你被招聘的机会等同榜首项。3).在许多的处理器/编译器中,浮点一般都是不行重入的。有些处理器/编译器需要让额处的寄存器入栈,有些处理器/编译器便是不允许在ISR中做浮点运算。此外,ISR应该是短而有功率的,在ISR中做浮点运算是不明智的。4).与第三点一脉相承,printf()常常有重入和功能上的问题。假如你丢掉了第三和第四点,我不会太尴尬你的。不必说,假如你能得到后两点,那么你的被招聘远景越来越光明晰。ISR不调用print函数。12、下面的代码输出是什么,为什么?
voidfoo(void){unsignedinta=6;intb=-20;(a+b>6)?puts(“>6”):puts(“<=6”);}这个问题检验你是否懂得C言语中的整数自动转化准则,我发现有些开发者懂得很少这些东西。不论怎么,这无符号整型问题的答案是输出是“>6”。原因是当表达式中存在有符号类型和无符号类型时一切的操作数都自动转化为无符号类型。因而-20变成了一个十分大的正整数,所以该表达式计算出的成果大于6。这一点关于应当频频用到无符号数据类型的嵌入式体系来说是丰常重要的。假如你答错了这个问题,你也就到了得不到这份作业的边缘。13、评价下面的代码片断:
unsignedintzero=0;unsignedintcompzero=0xFFFF;/*1’scomplementofzero*/关于一个int型不是16位的处理器为说,上面的代码是不正确的。应编写如下:
unsignedintcompzero=~0;这一问题真实能揭露出应试者是否懂得处理器字长的重要性。在我的经历里,好的嵌入式程序员十分准确地明白硬件的细节和它的局限,但是PC机程序往往把硬件作为一个无法防止的烦恼。到了这个阶段,应试者或许彻底垂头丧气了或许信心满满志在必得。假如明显应试者不是很好,那么这个检验就在这儿完毕了。但假如明显应试者做得不错,那么我就扔出下面的追加问题,这些问题是比较难的,我想只是十分优秀的应试者能做得不错。提出这些问题,我希望更多看到应试者应付问题的办法,而不是答案。不论怎么,你就当是这个娱乐吧…14、虽然不像非嵌入式计算机那么常见,嵌入式体系仍是有从堆(heap)中动态分配内存的过程的。那么嵌入式体系中,动态分配内存或许产生的问题是什么?这儿,我希望应试者能说到内存碎片,碎片搜集的问题,变量的持行时刻等等。这个主题现已在ESP杂志中被广泛地评论过了(主要是P.J.Plauger,他的解说远远超过我这儿能说到的任何解说),一切回过头看一下这些杂志吧!让应试者进入一种虚伪的安全感觉后,我拿出这么一个小节目:下面的代码片段的输出是什么,为什么?
Char*ptr;if((ptr=(char*)malloc(0))==NULL)puts(“Gotanullpointer”);elseputs(“Gotavalidpointer”);这是一个有趣的问题。最近在我的一个同事不经意把0值传给了函数malloc,得到了一个合法的指针之后,我才想到这个问题。这便是上面的代码,该代码的输出是“Gotavalidpointer”。我用这个来开端评论这样的一问题,看看被面试者是否想到库例程这样做是正确。得到正确的答案当然重要,但解决问题的办法和你做决定的根本原理更重要些。15、Typedef在C言语中频频用以声明一个现已存在的数据类型的同义字。也能够用预处理器做相似的事。例如,考虑一下下面的比如:
#definedPSstructs*typedefstructs*tPS;以上两种状况的意图都是要界说dPS和tPS作为一个指向结构s指针。哪种办法更好呢?(假如有的话)为什么?这是一个十分奇妙的问题,任何人答对这个问题(正当的原因)是应当被祝贺的。答案是:typedef更好。考虑下面的比如:
dPSp1,p2;tPSp3,p4;榜首个扩展为structs*p1;structsp2;上面的代码界说p1为一个指向结构的指,p2为一个实践的结构,这或许不是你想要的。第二个比如正确地界说了p3和p4两个指针。16、C言语赞同一些令人震惊的结构,下面的结构是合法的吗,假如是它做些什么?
inta=5,b=7,c;c=a+++b;这个问题将做为这个检验的一个愉快的结束。不论你相不相信,上面的比如是彻底符合语法的。问题是编译器怎么处理它?水平不高的编译作者实践上会争论这个问题,根据最处理准则,编译器应当能处理尽或许一切合法的用法。因而,上面的代码被处理成:
c=a+++b;因而,这段代码持行后a=6,b=7,c=12。假如你知道答案,或猜出正确答案,做得好。假如你不知道答案,我也不把这个当作问题。我发现这个问题的最大好处是:这是一个关于代码编写风格,代码的可读性,代码的可修正性的好的论题。

未经允许不得转载:IT技术网站 » c语言基础知识(题库)
分享到: 更多 (0)

评论 抢沙发

评论前必须登录!

 

志在指尖 用双手敲打未来

登录/注册IT技术大全

热门IT技术

C#基础入门   SQL server数据库   系统SEO学习教程   WordPress小技巧   WordPress插件   脚本与源码下载