1.关于指针?
指针虽然强大,与之相伴的风险却也不小。如果使用得当,它们可以简化算法的实现,并使其更富效率;如果使用不当,它们会引起错误,导致细微而令人困惑的症状,并且极难发现原因。
2.为什么学习C语言?
(1)效率高
a.优秀C程序的效率几乎和汇编语言程序一样高,但C程序明显比汇编语言程序更易于开发
b.和其他更多语言相比,C给予程序员更多的控制权,如控制数据的存储位置和初始化过程等
c.C缺乏“安全网”特性,这虽有助于它的效率,但也增加了出错的可能性。例如,C对数组下标引用和指针访问并不进行有效性检查,这可以节省时间,但你在使用这些特性时必须特别小心。
启示:在使用C语言时要严格遵守相关规定,这样就可以避免这些潜在的问题。
(2)可直接访问硬件
C提供了丰富的操作符集合,它们可以让程序员有效地执行一些底层的计算如移位和屏蔽等,而不必求助于汇编语言。C的这个特点使很多人把C称为“高级”的汇编语言。但是,当需要的时候,C程序可以很方便地提供汇编语言的接口。这些特性使C成为实现操作系统和嵌入式控制器软件的良好选择。
(3)ANSI标准提高了C程序在不同机器上的可移植性。
(4)C是C++的基础。
3.C语言的两种标准?
K&R C和ANSI C
ANSI C是K&R C的后继者,并已在根本上取代了后者。
4.排版说明
“K&R C” 表示我正在讨论ANSI C和K&R C之间的重要区别。尽管绝大数以K&R C写成的程序仅需极微小的修改即可在ANSI C环境运行,但有时你仍可能遇到一个ANSI之前的编译器,或者遇到一个更老式的程序。如此一来,两者的区别便至关重要。
5.注释
在C语言中,要从逻辑上删除一段C代码,更好的方法是使用#if指令(而不是/和/符号来注释)。
if 0
statements
endif
在#if和#endif之间的程序段就可以有效地从程序中去除,即使这段代码之间原先存在注释也无妨,所以这是一种更为安全的方法。
6.预处理指令
预处理指令是由 预处理器 解释的。预处理器读入源代码,根据预处理指令对其进行修改,然后把修改过的源代码交给编译器。
在我们地例子中,预处理器用名为stdio.h的库函数头文件的内容替换第1条#include指令语句,其结果就仿佛是stdio.h的内容被逐字写到源文件的那个位置。
提示:如果你有一些声明需要用于几个不同的源文件,这个技巧也是一种方便的方法-------你在一个单独的文件中编写这些声明,然后用#include指令把这个文件包含到需要使用这些声明的源文件中。
问题:为什么程序开头要有函数原型?
答案:当这些函数被调用时,编译器就能对它们进行准确性检查。
第2个和第4个参数被声明为const,这表示函数将不会修改函数调用者所传递的这两个参数。
提示:
假如这个程序的源代码由几个源文件所组成,那么使用该函数的源文件都必须写明该函数的原型。把原型放在头文件中并使用#include指令包含它们,可以避免由于同一个声明的多份拷贝而导致的维护性问题。
7.main函数
所有4个变量都是main函数的局部变量,其他函数都不能根据它们的名字访问它们。当然,它们可以作为参数传递给其他函数。
事实上。关于C函数的参数都是按值传递的。
但是,当数组名作为参数时就会产生按引用传递的效果。
8.gets函数
gets函数从标准输入读取一行文本并把它存储于作为参数传递给它的数组中。一行输入由一串字符组成,以一个换行符结尾。gets函数丢弃换行符,并在该行的末尾存储一个NULL字节(一个NULL字节是指字节模式为全0的字节,类似'\0'这样的字符常量)。然后,gets函数返回一个非NULL值,表示该行已经被成功读取。当gets函数被调用但事实上不存在输入行时,它就返回NULL值,表示它达到了输入的末尾(文件尾)。
9.NULL
NULL是ANSII字符集中'\0'字符的名字,它的字节模式为全0。NULL也指一个其值为0的指针。不同的地方,含义也不同。
10.字符串就是一串以NULL字节结尾的字符。
11.函数的起始部分跟程序中的函数原型必须完全匹配。
在函数声明的数组参数中,并未指定数组长度。这种格式是正确的,因为不论调用函数的程序传递给它的数组参数的长度是多少,这个函数都照收不误。这是一个伟大的特性,它允许单个函数操作任意长度的一维数组。这个特性不利的一面是函数没法知道该数组的长度。如果确实需要数组的长度,它的值必须作为一个单独的参数传递给函数。
和绝大多数语言一样,C语言中形式参数的名字和实际参数的名字并没有什么关系。你可以让两者相同,但这不是必须的。
局部变量未初始化,那这个变量的初始值是一个不可预料的值,也就是垃圾。
提示:
标准并未硬性规定C编译器对数组下标的有效性记性检查,而且绝大多数C编译器确实也不进行检查。因此,如果你需要进行数组下标的有效性检查,你必须自行编写代码。如果此处不进行num<max这个测试,而且程序所读取的文件包含超过20个列标号,那么多出来的值就会存储在紧随数组之后的内存位置,这样就会破坏原先存储在这个位置的数据,可能是其他变量,也可能是函数的返回地址。这可能会导致多种结果,程序很可能不会按照你预想的那样运行。
警告:
用于测试两个表达式是否相等的操作符是==。如果误用了=操作符,虽然它也是一个合法的表达式,但结果几乎肯定和你的本意不一样。但由于它也是一个合法的表达式,所以编译器无法为你找出这个错误(有些较新的编译器在发现if和while表达式中使用赋值符时会发出警告信息,其理论是在这样的上下文环境中,用户需要使用比较操作的可能性要远大于赋值操作)。
12.数组就是一段内存单元,所以,数组中存储的也是二进制0,1序列。
13.暂未看懂的部分?
while((ch=getchar())!=EOF && ch!='\n')
;
14.可以把程序:
ch=getchar();
while(ch!=EOF && ch!='\n')
ch=getchar();
改写为:
while((ch=getchar())!=EOF && ch!='\n')
;
原程序两次出现ch=getchar();这条语句
C可以把赋值操作蕴含于while语句内部,这样就允许程序员消除冗余语句。
问题:
为什么ch被声明为整型,而我们事实上需要它来读取字符?
答案:
EOF是一个整型值,它的位数比字符类型要多,把ch声明为整型可以防止从输入读取的字符意外地被解释为EOF。但同时,这也意味着接收字符的ch必须足够大,足以容纳EOF,这就是ch使用整型值的原因(字符只是小整型数而已,所以用一个整型值变量容纳字符值并不会引起任何问题)。
15.正因为实际传递的是一个指针而不是一份数组的拷贝,才使数组名作为参数时具备了传址调用的语义。
16.例子程序把columns声明为const就有两方面的作用。首先,它声明该函数的坐着的意图是这个参数不能被修改。其次,它导致编译器去验证是否违背该意图。