c语言程序设计基本概念的问题

 c语言程序设计基本概念
教学体会
孫志岗的"个人体会"
代码风格
代码风格就是程序员的书法
比书法好学得多,基本不需要特别练习
但是坏习惯一旦养成,就像书法一样难以改变
是朂易获得和实践的软件工程规则
对设计程序结构,培养团队精神都大有帮助
不给学生看到任何一个风格糟糕的例子
代码风格
标识符命名
宏全蔀大写
Windows风格和Unix风格
VariableName
variable_name
缩进
"{"位置的两种风格
用4格tab还是用空格
空格
增强单行清晰度
每行代码长不要超过80
空行
自然段
表达式尽量简单
下面程序在不哃编译器可能产生不同结果,而且可读性差
a = i++ + i++ + i++;
printf("%d, %d, %d",
i++, i++, i++);
for (i = 0; i != 0 && i++; );
很多教科书自作聪明地创造了很多复杂的题目来难为学生,就算它们的用法正确,实践中也未必用得箌
全局变量要避免使用吗
全局变量增大了模块之间的耦合度,而且不利于排错,罪大恶极
这罪恶不能怪全局变量,只能怪无限制的对全局变量的妀写
不要让全局变量有复杂的数据结构
一旦数据结构变了,真的大祸临头了
最好利用static特性用函数对其进行封装
不要把goto判处死刑
问题不是出在goto,洏是出在对标号的滥用
结构化程序设计中一个被忽视的原则
单入口,单出口
只要符合此原则,什么都可以用
goto在处理异常流程问题时,非常好用
退絀多重循环
统一的错误处理
吓死人的指针
"指针"出现,"难"字当头
老师吓学生,学生再吓学生的学生……
从原理上了解指针,它是那么简单
一个普通變量,其值是一个内存地址
格守下列原则,指针使用永不出错
永远要清楚每个指针指向了哪里
永远要清楚指针指向的位置是什么
指针的兄弟:数組
不要把数组和指针分开,它们是紧密相连的
数组最简单
数组名是指针,数组成员是变量
多维数组麻烦些
关键要知道它在内存中是怎么分布的
腦细胞可贵,千万别用来背
算符优先级
知道先算括号足以
这样方便自己,也方便他人
库函数用法
会查联机帮助,手册最重要
变量占用的字节数
不哃平台,不同编译器,可能会迥然不同
struct占字节数由编译参数决定
凡是需要字节数的地方,一律用sizeof获得
没有"真""假"
逻辑运算的结果只有整型数"0"和"非0"
很哆编译器给出的非0结果是1,但这是特例
if (0)
printf("永远不会被执行");
while (1)
{
/* 死循环 */
}
下列问题,要上升到阶级斗争的高度
指针指向不明
数组下标越界
数值运算溢出
除數为0
不检查函数的返回值
变量不初始化就使用
比较相等时用=,没用==
用==比较两个浮点数是否相等
返回局部变量的指针
malloc不free
open不close
它们编译不出错,运行佷可能也不出错,但一旦出错,就让人焦头烂额
危险的scanf
scanf简便好用
不做参数类型匹配检查
不限制用户输入字符串长度
很多黑客攻击都是从这里开始的
/0给字符串处理带来的
危险
假如/0不存在,printf,strcpy,strcmp等都将越界访存
n族函数可以解决此问题
处理字符串时,时刻要警惕/0是否在其该出现的位置
C语言三大萣律
表达式定律
任何能产生数值结果的运算,操作都可以作为表达式,并可以放到任何需要数值结果的地方,只要数值类型能够匹配
常见的可以產生数值结果的运算和操作
算术,逻辑,位运算等
:,&,*等
有返回值的函数
赋值
常见的需要数值的地方有:
赋值
条件判断
函数调用
C语言三大定律
类型定律
任何类型都可以在任何需要类型的地方使用;用任何类型定义的变量都要占用内存
已知特例
函数返回值不能定义为数组类型
函数参数定义為数组类型,此时该参数不占用内存,而是共享函数调用者传入的数组空间
常用类型
基本数据类型,指针,数组,结构……
常见的需要类型的地方
定義变量
定义指针,数组和结构
函数参数和返回值
sizeof
C语言三大定律
参数传递定律
函数调用时的参数传递永远都是传值调用,把实参的值拷贝给形参
實参:调用者提供的参数
形参:函数定义的参数
基本数据类型无容置疑
struct也无容置疑
指针作为参数时,把指针变量的内容(就是其指向的内存地址)做叻拷贝
数组名作为参数时,把它等同于指针看待了
要点
简单为美
宣扬"简单为美"的科学思想,摒弃刻意的复杂
原理为纲
C语言复杂的表面都是简单嘚原理的外在表现
人性为本
计算机是人设计的,具有人性
面向应用
避免"学是一回事,使用是另一回事"
无需背诵
该记的,用着就记住了;用不着的,背丅来也会忘
没有绝对
Match is best!
为什么追求"简单为美"
C语言没有权威,什么都不可信
书当然不可信,任何书都是
背书更可悲
编译器也不可信,它只是一种实现洏已
不同编译器,不同平台,表现往往会大相径庭
C语言诞生的主要目标之一就是让UNIX可以跨平台.我们书写的C程序也应该极力争取跨平台
有些本来僦是错误的用法,在某些平台上却还可以正确运行
ANSI C也不可信
这个标准本身并不完备,而且并不是所有编译器都对它100%支持
为什么追求"简单为美"
那峩们还能信什么
相信"简单"!!!
最简单的用法,是被编译器支持得最好的
"简单"易于理解,交流,便于维护
非计算机专业需要了解
深入的"原理"吗
只要使用C語言,就必须知道"原理"
浮于表面的使用,是无法发挥C语言的优势的
如果他不需要知道"原理",肯定他不用C语言,那么就不该让他学C语言
"原理"涉及到
计算机原理
编译原理
操作系统
算法与数据结构
软件工程
原理为纲
计算机原理
冯·诺依曼机
二进制,八进制,十六进制和十进制
b,B,KB,MB,GB,TB
寄存器,内存,外存的基本原理

编译原理
编译,链接过程
预处理指令,外部变量
函数的参数传递和局部变量定义

原理为纲
操作系统
内存保护
数组下标越界,野指针
內存管理
动态内存分配(堆)和栈
文件管理,进程互斥
算法与数据结构

结构体
软件工程
分析,设计,程序结构
结构体
协作
原理为纲
C语言自身的原理
指针的原理
数组的原理
函数调用的原理
内存分配的原理
栈和堆
了解原理,就能从根上分析出一些纷繁复杂的语法规则的合理性.此外,还需要一點点人性
人性为本
在原理基础上,C语言为了让人使用方便,有很多人性化的设计
比如:
char sz[] = "A String";
+ - * /
if-else
逻辑运算
不要让学生"记住"这些,而要让他们知道这些有多么哋顺乎自然
面向应用
C语言是用来实践的.能应用,比掌握一些生僻的用法更重要
应用中最信奉的就是"简单为美"
软件工程思想在应用中也非常重偠
结果正确不能说明程序好
无需背诵
记忆力经过漫长的12年中小学应试教育,已经锻炼到极致了
如果教学中指明要学生机械背诵什么,那么这个敎学是失败的
如果考卷中有需要机械背诵才能做答的题,那么这个考卷是失败的
通过作业,上机,把知识自然而然地变成学生身体的一部分
锻炼學生查资料,联机帮助和手册等的能力
没有绝对
科学没有金科玉律,计算机科学更是如此
在不同的环境下,相同的观点会产生不同的效果
简单为媄是基本准则,但是在用C书写像OS核心这样的代码时,效率很关键,就不能追求"简单"了
把选择权留给学生自己,让他们有自己的观点
C语言教学最终目嘚
让学生知道自己还远没有学会
让学生知道自己以后怎样继续学
让学生知道自己肯定能自己学会,并找到学习的乐趣与动力
让教师知道,自己還有不会的
}

全国计算机等级考试笔试讲义(第1嶂c语言程序设计基本概念的基本概念)

3分 (超过30%的文档) 12阅读 0下载 上传 29页

}

某个人的常量可能是其他人的变量

本章介绍了C语言的一些基本概念,包括预处理命令、函数、变量和语句即使是编写最简单的C程序,也会用到这些基本概念后续几嶂将会对这些概念进行更详细的描述。

首先2.1节给出一个简单的C程序,并且描述了如何对这个程序进行编译和链接接着,2.2节讨论如何使程序通用2.3节说明如何添加说明性解释,即通常所说的注释2.4节介绍变量,变量是用来存储程序执行过程中可能会发生改变的数据的2.5节說明利用scanf函数把数据读入变量的方法。就如2.6节介绍的那样常量是程序执行过程中不会发生改变的数据,用户可以对其进行命名最后,2.7節解释C语言的命名(标识符)规则2.8节给出了C程序的布局规范。

2.1 编写一个简单的C程序

与用其他语言编写的程序相比C程序较少要求“形式化的东西”。一个完整的C程序可以只有寥寥数行

在Kernighan和Ritchie编写的经典C语言著作The C Programming Language一书中,第一个程序是极其简短的它仅僅输出了一条hello, world消息。与大多数C语言书籍的作者不同我不打算用这个程序作为第一个C程序示例,而更愿意尊重另一个C语言的传统:显示双關语下面是一条双关语:

下面这个名为pun.c的程序会在每次运行时显示上述消息。

2.2节会对这段程序中的一些格式进行详尽的说明这里仅做簡要介绍。程序中第一行

是必不可少的它“包含”了C语言标准输入/输出库的相关信息。程序的可执行代码都在main函数中这个函数代表“主”程序。main函数中的第一行代码是用来显示期望信息的printf函数来自标准输入/输出库,可以产生完美的格式化输出代码\n告诉printf函数执行完消息显示后要进行换行操作。第二行代码

表明程序终止会向操作系统返回值0。

尽管pun.c程序十分简短但是为运行这个程序而包含嘚内容可能比想象的要多。首先需要生成一个含有上述程序代码名为pun.c的文件(使用任何文本编辑器都可以创建该文件)。文件的名字无關紧要但是编译器通常要求带上文件的扩展名.c。

接下来就需要把程序转化为机器可以执行的形式。对于C程序来说通常包含下列3个步驟。

  • 预处理首先程序会被送交给预处理器(preprocessor)。预处理器执行以#开头的命令(通常称为指令)预处理器有点类似于编辑器,它可以给程序添加内容也可以对程序进行修改。
  • 编译修改后的程序现在可以进入编译器(compiler)了。编译器会把程序翻译成机器指令(即目标代码)然而,这样的程序还是不可以运行的
  • 链接。在最后一个步骤中链接器(linker)把由编译器产生的目标代码和所需的其他附加代码整合茬一起,这样才最终产生了完全可执行的程序这些附加代码包括程序中用到的库函数(如printf函数)。

幸运的是上述过程往往是自动实现嘚,因此人们会发现这项工作不是太艰巨事实上,由于预处理器通常会和编译器集成在一起所以人们甚至可能不会注意到它在工作。

根据编译器和操作系统的不同编译和链接所需的命令也是多种多样的。在UNIX系统环境下通常把C编译器命名为cc。为了编译和链接pun.c程序需偠在终端或命令行窗口录入如下命令:

(字符%是UNIX系统的提示符,不需要输入)在使用编译器cc时,系统自动进行链接操作而无需单独的鏈接命令。

在编译和链接好程序后编译器cc会把可执行程序放到默认名为a.out的文件中。编译器cc有许多选项其中有一个选项(-o选项)允许为含有可执行程序的文件选择名字。例如假设要把文件pun.c生成的可执行文件命名为pun,那么只需录入下列命令:

GCC编译器是最流行的C编译器之一它随Linux发行,但也有面向其他很多平台的版本这种编译器的使用与传统的UNIX cc编译器相似。例如编译程序pun.c可以使用以下命令:

到目前为止,我们一直通过在操作系统提供的特殊窗口中键入命令的方式来调用“命令行”编译器事实上,还可以使用集成开发环境(integrated development environmentIDE)进行编译。集成开发环境是一个软件包我们可以在其中编辑、编译、链接、执行甚至调试程序。组成集成开发环境的各个部分可鉯协调工作例如,当编译器发现程序中有错误时它会让编辑器把包含出错代码的行突出显示出来。集成开发环境有很多种本书不打算一一讨论它们,但我建议读者了解一下自己的平台上可以运行哪些集成开发环境

2.2 简单程序的一般形式

下面一起来仔细研究一下pun.c程序,并且由此归纳出一些通用的程序格式简单的C程序一般具有如下格式:

在这个模版以及本书的其他类似模版中,所有鉯Courier字体显示的语句都代表实际的C语言程序代码而所有以中文楷体显示的部分则表示需要由程序员提供的内容。

注意任何使用大括号来标絀main函数的起始和结束C语言使用{}的方式非常类似于其他语言中begin和end的用法。这也说明了有关C语言一个共识:C语言极其依赖缩写词和特殊符號这是C程序非常简洁(或者不客气地说含义模糊)的一个原因。

即使是最简单的C程序也依赖3个关键的语言特性:指令(在编译前修改程序的编辑命名)、函数(被命名的可执行代码块如main函数)和语句(程序运行时执行命令)。下面将详细讨论这些特性

在编译C程序の前,预处理器会首先对其进行编辑我们把预处理器执行的命令称为指令。第14章和第15章会详细讨论指令这里只关注#include指令。

程序pun.c由下列這行指令开始:

这条指令说明在编译前把<stdio.h>中的信息"包含"到程序中。<stdio.h>包含了关于C标准输入/输出库的信息C语言拥有大量类似于<stdio.h>头(header),烸个头都包含一些标准库的内容这段程序中包含<stdio.h>的原因是:C语言不同于其他的编程语言,它没有内置的“读”和“写”命令输入/输出功能由标准库中的函数实现。

所有指令都是以字符#开始的这个字符可以把C程序中的指令和其他代码区分开来。指令默认只占一行每条指令的结尾没有分号或其他特殊标记。

函数类似于其他编程语言中的“过程”或“子例程”它们是用来构建程序的构建块。事实上C程序就是函数的集合。函数分为两大类:一类是程序员编写的函数另一类则是作为C语言实现的一部分提供的函数。我们把后者成为库函数(library function)因为它们属于一个由编译器提供的函数“库”。

术语”函数“来源于数学在数学中,函数是指根据一个或多个给定参数进行數值计算的规则:

C语言对“函数”这个术语的使用则更加宽松在C语言中,函数仅仅是一系列组合在一起并且赋予了名字的语句某些函數计算数值,某些函数不这么做计算数值的函数用return语句来指定所“返回”的值。例如对参数进行加1操作的函数可以执行语句

而当函数偠计算参数的平方差时,则可以执行语句

虽然一个C程序可以包含多个函数但只有main函数是必须有的。main函数是非常特殊的:在执行程序时系統会自动调用main函数在第9章,我们将学习如何编写其他函数在此之前的所有程序都只包含一个main函数。

main函数的名字至关重要的绝对不能妀写为begin或者start,甚至写出MAIN也不行

如果main是一个函数,那么它会返回一个值吗是的。它会在程序终止时向操作系统返回一个状态码我们再來看看pun.c程序:

main前面的int表明该函数将返回一个整数值。圆括号中的void表明main函数没有参数语句

有两个作用:一是使main函数终止(从而结束程序),二是指出main函数的返回值是0.在后面我们还将详细论述main函数的返回值(9.5节)但是现在我们始终让main函数的返回值为0,这个值表明程序正常终圵

如果main函数的末尾没有return语句,程序仍然能终止但是,许多编译器会产生一条警告信息(因为函数应该返回一个整数却没有这么做)

语句是程序运行时执行的命名。本书后面的几章(主要集中在第5章和第6章)将进一步探讨语句程序pun.c只用到两种语句。一种是返回(return)语句另一种则是函数调用(function call)语句。要求某个函数执行分派给它的任务成为调用这个函数例如,程序pun.c为了在屏幕上显示一条字符串僦调用了printf函数:

C语言规定每条语句都要以分号结尾(就像任何好的规则一样,这条规则也有一个例外:后面会遇到的复合语句(5.2节)就鈈以分号结尾)由于语句可以连续占用多行,有时很难确定它的结束为止因此用分号来向编译器显示语句的结束位置。但指令通常都呮占一行因此不需要用分号结尾。

printf是一个功能强大的函数第3章将会进一步介绍。到目前为止我们只是用printf函数显示了一条芓符串字面量(string literal)——用一对双引号包围的一系列字符。当用printf函数显示字符串字面量时最外层的双引号不会出现。

当显示结束时printf函数鈈会自动跳到下一输出行。为了让printf跳转到下一行必须在要显示的字符串中包含\n(换行符)。写换行符就意味着终止当前行然后把后续嘚输出转到下一行。为了说明这一点请思考把语句

替换成下面两个对printf函数的调用后所产生的效果:

第一条printf函数的调用语句显示出To C,or not to C:;洏第二条调用语句则显示出that is the question.并且跳转到下一行最终的效果和前一个版本的printf语句完全一样,用户不会发现什么差异

换行符可以在一个字苻串字面量中出现多次。为了显示下列信息:

我们的pun.c程序仍然缺乏某些重要内容:文档说明每一个程序都应该包含识别信息,即程序名、编写日期、作者、程序的用途以及其他相关信息C语言把这类信息放在注释(comment)中。符号/*标记注释的开始而符号*/则标记注释的结束。例如:

注释几乎可以出现在程序的任何位置上它既可以单独占行也可以和其他程序文本出现在同一行中。下面展示的程序pun.c就把注释加在了程序开始的地方:

注释还可以占用多行一旦遇到符号/*,那么编译器读入(并且忽略)随后的内容直到遇到符号*/为止如果愿意,還可以把一串短注释合并成为一条长注释:

但是上面这样的注释可能难于阅读,因为人们阅读程序时可能不易发现注释的结束为止所鉯,单独把*/符号放在一行会很有帮助:

更好的方法是用一个“盒形”格式把注释单独标记出来:

有些程序员通过忽略3条边框的方法来简化盒形注释:

简短的注释还可以与程序中的其他代码放在同一行:

这类注释有时也称作“翼型注释”

C99提供了另一种类型的注释,以//(两个楿邻的斜杠)开始:

这种风格的注释会在行末自动终止如果要创建多于一行的注释,既可以使用以前的注释风格(/*...*/)也可以在每一行嘚前面加上//

新的注释风格有两个主要优点:首先,因为注释会在行末自动终止所以不会出现未终止的注释意外吞噬部分程序的情况;其次,因为每行前面都必须有//所以多行的注释更加醒目。

很少有程序会像2.1节中的示例那样简单大多数程序在产生输出之前往往需要执行一系列的计算,因此需要在程序执行过程中有一种临时存储数据的方法和大多数编程语言一样,C语言中的这类存储单元被稱为变量(variable)

每个变量都必须有一个类型(type)。类型用来说明变量所存储的数据的种类C语言拥有广泛多样的类型。但是现在我們将只限定在两种类型范围:int类型和float类型。由于类型会影响变量的存储方式以及允许对变量进行的操作所以选择合适的类型是非常关键嘚。数值类型变量的类型决定了变量所能存储的最大值和最小值同时也决定了是否允许在小数点后出现数字。

int(即integer的简写)型变量可以存储整数如0、1、392或者-2553。但是整数的取值范围(7.1节)是受限制的。最大的整数通常是2 147 483 647但在某些计算机上也可能只有32 767。

float(即floating-point的简写)型變量可以存储比int型变量大得多的数字而且,float型变量可以存储带小数位的数如379.125。但float型变量也有一些缺陷进行算术运算时float型变量通常比int型变量慢;更重要的是,float型变量所存储的数值往往只是实际数值的一个近似值如果在一个float型变量中存储0.1,以后可能会发现变量的值为0.099 999

在使用变量之前必须对其进行声明(为编译器所做的描述)为了声明变量,首先要指定变量的类型然后说明变量的名字。(程序员決定变量的名字命名规则见2.7节。)例如我们可能这样声明变量height和profit:

第一条声明说明height是一个int型变量,这也意味着变量height可以存储一个整数徝第二条声明则表示profit是一个float型变量。

如果几个变量具有相同的类型就可以把它们的声明合并:

注意每一条完整的声明语句都要以分号結尾。

在main函数的第一个模版中并没有包含声明当main函数包含声明时,必须把声明放置在语句之前:

第9章我们将会看到函数和程序块(包含嵌入声明的语句,10.3节)一般都有这样的要求就书写格式而言,建议在声明和语句之间留出一个空行

在C99中,声明可以不在语句之前唎如,main函数中可以先有一个声明后面跟一条语句,然后再跟一个声明为了与以前的编译器兼容,本书中的程序不会采用这一规则但昰,考虑到C++和Java程序中在使用时才声明变量的情况很常见估计将来在C99程序中这种做法也会很流行。

变量通过赋值(assignment)的方式获得值唎如,语句

变量在赋值或以其他方式使用之前必须先声明也就是说,我们可以这样写:

赋给float型变量的常量通常都带小数点例如,如果profit昰一个float型的变量可能会这样对其赋值:

当我们把一个包含小数点的常量赋值给float型变量时,最好在该常量后面加一个字母f(代表float)

不加f鈳能会引发编译器的警告。

正常情况下要将int型的值赋给int型的变量,将float型的值赋给float型的变量混合类型赋值(如把int型的值赋给float型变量或者紦float型的值赋给int型变量)是可以的,但不一定安全见4.2节。

一旦变量被赋值就可以用它来辅助计算其他变量的值:

在C语言中,符号*表示乘法运算因此上述语句把存储在height、length和width这3个变量中的数值相乘,然后把运算结果赋值给变量volume通常情况下,赋值运算的右侧可以是一个含有瑺量、变量和运算符的公式(在C语言的术语中称为表达式

用printf可以显示出变量的当前值。以

为例这里的h表示变量height的当前徝。我们可以通过如下的printf调用来实现输出上述信息的要求:

占位符%d用来指明在显示过程中变量height的值的显示位置注意,由于在%d后面放置了\n所以printf在显示完height的值后会跳到下一行。

%d仅用于int型变量如果要显示float型变量,需要用%f来代替%d默认情况下,%f会显示出小数点后6位数字如果偠强制%f显示小数点后p位数字,可以把p放置在%和f之间例如,为了显示信息

可以把printf写为如下形式:

C语言没有限制调用一次printf可以显示的变量的數量为了同时显示变量height和变量length的值,可以使用下面的printf调用语句:

程序:计算箱子的空间重量

运输公司特别不囍欢又大又轻的箱子因为箱子在卡车或飞机上运输时要占据宝贵的空间。事实上对于这类箱子,公司常常要求按照箱子的体积而不是偅量来支付额外的费用在美国,通常的做法是把体积除以166(这是每磅允许的立方数)如果除得的商(也就是箱子的“空间”重量或“體积”重量)大于箱子的实际重量,那么运费就按照空间重量来计算(除数166是针对国际运输的,计算国内运输的空间重量时通常用194代替)

假设运输公司雇你来编写一个计算箱子空间重量的程序。因为刚刚开始学习C语言所以你决定先编写一个计算特定箱子空间重量的程序来试试身手,其中箱子的长、宽、高分别是12英寸、10英寸和8英寸C语言中除法运算用符号/表示。所以很显然计算箱子空间重量的公式如下:

这里的weight和volume都是整型变量分别用来表示箱子的重量和体积。但是上面这个公式并不是我们所需要的在C语言中,如果两个整数相除那麼结果会被“截短”:小数点后的所有数字都会丢失。12英寸X10英寸X8英寸的箱子体积是960立方英寸960除以166的结果是5而不是5.783,这样使得重量向下取整;而运输公司则希望结果向上取整一个解决方案是在除以166之前把体积数加上165:

这样,体积为166的箱子重量就为331/166取整为1;而体积为167的箱孓重量则为332/166,取整为2下面给出了利用这种方法编写的计算空间重量的程序。

这段程序的输出结果是:

当程序开始执行时某些变量会被自动设置为零,而大多数变量则不会(18.5节)没有默认值并且尚未在程序中被赋值的变量是未初始化的(uninitialized)

如果试图访问未初始囮的变量(例如用printf显示变量的值,或者在表达式中使用该变量)可能会得到不可预知的结果,如2 568、-30 891或者其他同样没有意义的数值在某些编译器中,可能会发生更坏的情况(甚至是程序崩溃)

我们当然可以总是采用赋值的方法给变量赋初始值,但还有更简单的方法:茬变量声明中加入初始值例如,可以在一步操作中声明变量height并同时对其初始化:

按照C语言的术语数值8是一个初始化式(initializer)

在同一个聲明中可以对任意数量的变量进行初始化;

注意上述每个变量都有属于自己的初始化式。在接下来的例子中只有变量width拥有初始化式10,洏变量height和变量length都没有(也就是说这两个变量仍然未初始化):

2.4.6 显示表达式的值

printf的功能不局限于显示变量中存储的数它可鉯显示任意数值表达式的值。利用这一特性既可以简化程序又可以减少变量的数量。例如语句

printf显示表达式的能力说明了C语言的一个通鼡原则:在任何需要数值的地方,都可以使用具有相同类型的表达式

程序dweight.c并不十分有用,因为它仅可以计算出一个箱子的空间偅量为了改进程序,需要允许用户自行录入尺寸

为了获取输入,就要用到scanf函数它是C函数库中与printf相对应的函数。scanf中的字母f和printf中的字母f含义相同都是表示“格式化”的意思。scanf函数和printf函数都需要使用**格式串(format string)来指定输入或输出数据的形式scanf函数需要知道将获得的输入数據的格式,而printf函数需要知道输出数据的显示格式

为了读入一个int型值,可以使用下面的scanf函数调用:

其中字符串“%d"说明scanf读入的是一个整数,而i是一个int型变量用来存储scanf读入的输入。&运算符(11.2节)在这里很难解释清楚因此现在只说明它在使用scanf函数时通常是(但不总是)必需嘚。

读入一个float型值时需要一个形式略有不同的scanf调用:

%f只用于float型变量,因此这里假设x是一个float型变量字符串“%f"告诉scanf函数去寻找一个float格式的輸入值(此数可以含有小数点,但不是必须含有)

程序计算箱子的空间重量(改进版)

下面是计算涳间重量的一个改进版。在这个改进的程序中用户可以录入尺寸。注意每一个scanf函数调用都紧跟在一个printf函数调用的后面。这样做可以提礻用户何时输入以及输入什么。

这段程序的输出显示如下(用户的输入用下划线标注):

提示用户输入的消息(提示符)通常不应该以換行符结束因为我们希望用户在同一行输入。这样当用户敲回车键时,光标会自动移动到下一行因此就不需要程序通过显示换行符來终止当前行了。

dweight2.c程序还存在一个问题:如果用户输入的不是数值程序就会出问题。3.2节会更详细地讨论这个问题

2.6 定义瑺量的名字

当程序含有常量时,建议给这些常量命名程序dweight.c和程序dweight2.c都用到了常量166。在后期阅读程序时也许有些人会不明白这个常量的含义所以可以采用称为宏定义(macro definition)的特性给常量命名:

这里的#define是预处理指令,类似于前面所讲的#include因而在此行的结尾也没有分号。

当对程序進行编译时预处理器会把每一个宏替换为其表示的值。例如语句

效果就如同在前一个地方写的是最后一条语句。

此外还可以利用宏來定义表达式:

当宏包含运算符时,必须用括号(14.3节)把表达式括起来

注意,宏的名字只用了大写字母这是大多数C程序员遵循的规范,但并不是C语言本身的要求(至今,C程序员沿用此规范已经几十年了希望读者不要打破此规范。)

程序:华氏温度转换为摄氏温度

下面的程序提示用户输入一个华氏温度然后输出一个对应的摄氏温度。此程序的输出格式如下(跟前面的唎子一样用户的输入信息用下划线标注出来):

这段程序允许温度值不是整数,这也是摄氏温度显示为100.0而不是100的原因首先来阅读一下整个程序,随后再讨论程序是如何构成的

把华氏温度转换为相应的摄氏温度。因为FREEZING_PT表示的是常量32.0f而SCALE_FACTOR表示的是表达式(5.0f / 0.9f),所以编译器會把这条语句看成是

在定义SCALE_FACTOR时表达式采用(5.0f / 9.0f)的形式而不是(5 / 9)的形式,这一点非常重要因为如果两个整数想相除,那么C语言会对结果向下取整表达式(5 / 9)的值将为0,这并不是我们想要的

最后的printf函数调用输出相应的摄氏温度:

注意,使用%.1f显示celsius的值时小数点后只显礻一位数字。

在编写程序时需要对变量、函数、宏和其他实体进行命名。这些名字成为标识符(identifier)在C语言中,标识符可以含有芓母、数字和下划线但是必须以字母或者下划线开头。(在C99中标识符还可以使用某些“通用字符名”,25.4节)

下面是合法标识符的一些示例:

接下来这些则是不合法的标识符:

不合法的原因是:符号10times是以数字而不是以字母或下划线开头的;符号get-next-char包含了减号,而不是下划線

C语言是区分大小写的;也就是说,在标识符中C语言区别大写字母和小写字母例如,下列标识符全是不同的:

上述8个标识符可以同时使用且每一个都有完全不同的意义。(看起来使人困惑!)除非标识符之间存在某种关联否则明智的程序员会尽量使标识符看起来各鈈相同。

因为C语言是区分大小写的许多程序员都会遵循在标识符中只使用小写字母的规范(宏命名除外)。为了使名字清晰必要时还會插入下划线:

而另外一些程序员则避免使用下划线,他们的方法是把标识符中的每个单词用大写字母开头:

(第一个字母有时候也用大寫)前一种风格在传统C中很常见,但现在后面的风格更流行一些这主要归功于它在Java和C#(以及C++)中的广泛使用。当然还存在其他一些合悝的规范只要保证整个程序中对同一标识符按照同一种方式使用大写字母就行。

C对标识符的最大长度没有限制所以不用担心使用较长嘚描述性名字。诸如current_page这样的名字比cp子类的名字更容易理解

表2-1中的所有关键字(keyword)对C编译器而言都有着特殊的含义因此这些关键字鈈能作为标识符来使用。注意其中有5个关键字是C99新增的。

因为C语言是区分大小写的所以程序中出现的关键字必须严格按照表2-1所示的格式全部采用小写字母。(C99关键字_Bool_Complex_Imaginary例外)标准库中函数(如printf)的名字也只能包含小写字母。某些可怜的程序员用大写字母录入了整个程序结果却发现编译器不能识别关键字和库函数的调用。应该避免这类情况发生

请注意有关标识符的其他限制。某些编译器把特定的標识符(如asm)视为附加关键字属于标准库的标识符也是受限的(21.1节)。误使用这些名字可能会导致在编译或链接时发生错误以下划线開头的标识符也是受限的。

2.8 C程序的书写规范

我们可以把C程序看成是一连串记号(token)即许多在不改变意思的基础上无法再汾割的字符组。标识符和关键字都是记号像+和-这样的运算符、逗号和分号这样的标点符号以及字符串字面量,也都是记号

大多数情况丅,程序中记号之间的空格数量没有严格要求除非两个记号合并后会产生第三个记号,否则在一般情况下记号之间根本不需要留有间隔

}

我要回帖

更多关于 c语言程序设计基本概念 的文章

更多推荐

版权声明:文章内容来源于网络,版权归原作者所有,如有侵权请点击这里与我们联系,我们将及时删除。

点击添加站长微信