1、源程序的结构特点源程序的结构特点1.一个语言源程序可以由一个或多个源文一个语言源程序可以由一个或多个源文件组成。件组成。2.每个源文件可由一个或多个函数组成。每个源文件可由一个或多个函数组成。3.一个源程序不论由多少个文件组成,都有一个源程序不论由多少个文件组成,都有一个且只能有一个一个且只能有一个main函数,即主函数。函数,即主函数。4.源程序中可以有预处理命令源程序中可以有预处理命令(include命令命令仅为其中的一种仅为其中的一种),预处理命令通常应放在,预处理命令通常应放在源文件或源程序的最前面。源文件或源程序的最前面。5.每一个说明,每一个语句都必须以分号结每一个说明,每一个语句
2、都必须以分号结尾。但预处理命令,函数头和花括号尾。但预处理命令,函数头和花括号“”之后不能加分号。之后不能加分号。6.标识符,关键字之间必须至少加一个空格标识符,关键字之间必须至少加一个空格以示间隔。若已有明显的间隔符,也可不以示间隔。若已有明显的间隔符,也可不再加空格来间隔。再加空格来间隔。语言词汇语言词汇标识符关键字运算符分隔符常量注释符1.标识符标识符在程序中使用的变量名、函数名、标在程序中使用的变量名、函数名、标号等统称为标识符。除库函数的函数名号等统称为标识符。除库函数的函数名由系统定义外,其余都由用户自定义。由系统定义外,其余都由用户自定义。C规定,标识符只能是字母规定,标识符只能
3、是字母(AZ,az)、数字、数字(09)、下划线、下划线(_)组成的字组成的字符串,并且其第一个字符必须是字母或符串,并且其第一个字符必须是字母或下划线。下划线。2.关键字(保留字)关键字(保留字)关键字是由语言规定的具有特定意义的关键字是由语言规定的具有特定意义的字符串。用户定义的标识符不应与关键字字符串。用户定义的标识符不应与关键字相同。相同。语言的关键字分为以下几类:语言的关键字分为以下几类:(1)类型说明符类型说明符用于定义、说明变量、函数或其它数据结构的类型。如前面例题中用于定义、说明变量、函数或其它数据结构的类型。如前面例题中用到的用到的int,double等等(2)语句定义符语句
4、定义符 用于表示一个语句的功能。如用于表示一个语句的功能。如ifelse就是条件语句的语句定义符。就是条件语句的语句定义符。(3)预处理命令字预处理命令字用于表示一个预处理命令。如前面各例中用到的用于表示一个预处理命令。如前面各例中用到的include。3.运算符 语言中含有相当丰富的运算符。运算语言中含有相当丰富的运算符。运算符与变量,函数一起组成表达式,表示符与变量,函数一起组成表达式,表示各种运算功能。运算符由一个或多个字各种运算功能。运算符由一个或多个字符组成。符组成。4.分隔符分隔符 逗号:主要用在类型说明和函数参数表逗号:主要用在类型说明和函数参数表中,分隔各个变量。中,分隔各个变
5、量。空格:多用于语句各单词之间,作间隔空格:多用于语句各单词之间,作间隔符。在关键字,标识符之间必须要有一符。在关键字,标识符之间必须要有一个以上的空格符作间隔,否则将会出现个以上的空格符作间隔,否则将会出现语法错误。语法错误。例如把例如把inta;写成写成inta;C编译器会把编译器会把inta当成当成一个标识符处理,其结果必然出错。一个标识符处理,其结果必然出错。5.常量常量 C语言中使用的常量可分为数字常量、字语言中使用的常量可分为数字常量、字符常量、字符串常量、符号常量、转义字符符常量、字符串常量、符号常量、转义字符等多种。在后面章节中将专门给予介绍。等多种。在后面章节中将专门给予介绍
6、。6.注释符注释符 C语言的注释符是以语言的注释符是以“/*”开头并以开头并以“*/”结结尾的串。在尾的串。在“/*”和和“*/”之间的即为注释。之间的即为注释。程序编译时,不对注释作任何处理。注释可程序编译时,不对注释作任何处理。注释可出现在程序中的任何位置。注释用来向用户出现在程序中的任何位置。注释用来向用户提示或解释程序的意义。在调试程序中对暂提示或解释程序的意义。在调试程序中对暂不使用的语句也可用注释符括起来,使翻译不使用的语句也可用注释符括起来,使翻译跳过不作处理,待调试结束后再去掉注释符跳过不作处理,待调试结束后再去掉注释符。sizeof运算符运算符 单目运算符,用以计算操作数在内
7、存中占用的字单目运算符,用以计算操作数在内存中占用的字节数。节数。操作数可以是两种:操作数可以是两种:括在圆括号中的类型标识符括在圆括号中的类型标识符 一个表达式一个表达式 比如:比如:sizeof(char)、sizeof(Lnode)typedef:类型定义符类型定义符 C+允许程序员用关键字允许程序员用关键字typedef自己定自己定义一个类型名。义一个类型名。格式为:格式为:typedef类型符类型符名字;名字;类型符可以是系统提供的某一个类型,如类型符可以是系统提供的某一个类型,如int、float、struct等。等。结构体类型结构体类型结构是由基本数据类型构成的、并用一个标识符来
8、命名的各种变结构是由基本数据类型构成的、并用一个标识符来命名的各种变量的组合。量的组合。结构中可以使用不同的数据类型。结构中可以使用不同的数据类型。1.结构说明和结构变量定义结构说明和结构变量定义在在TurboC中中,结构也是一种数据类型结构也是一种数据类型,可以使用结构变量可以使用结构变量,因此因此,象其它类型的变量一样象其它类型的变量一样,在使用结构变量时要先对其定义。在使用结构变量时要先对其定义。定义结构变量的一般格式为定义结构变量的一般格式为:struct结构名结构名类型类型变量名变量名;类型类型变量名变量名;.结构变量结构变量;结构名是结构的标识符不是变量名。结构名是结构的标识符不是
9、变量名。struct string char name8;int age;char sex2;char depart20;float wage1,wage2,wage3,wage4;struct string person;构成结构的每一个类型变量称为结构成员构成结构的每一个类型变量称为结构成员,它象数组的元素一样它象数组的元素一样,但数组中元素是以下但数组中元素是以下标来访问的标来访问的,而而结构是按变量名字来访问成结构是按变量名字来访问成员的。员的。如果省略结构名如果省略结构名,则称之为无名结构则称之为无名结构,这种情这种情况常常出现在函数内部况常常出现在函数内部,用这种结构时前面用这种结
10、构时前面的例子变成的例子变成:structcharname8;intage;charsex2;chardepart20;floatwage1,wage2,wage3,wage4;Tianyr,Liuqi;typedef struct int a;char b;list;指针指针 指针,一直被认为是指针,一直被认为是C语言中的精华。只有掌握语言中的精华。只有掌握了指针,你才能说你学会了了指针,你才能说你学会了C语言。语言。指针是一个特殊的变量,它里面存储的数值被解指针是一个特殊的变量,它里面存储的数值被解释成为内存里的一个地址。释成为内存里的一个地址。要搞清一个指针需要搞清指针的四方面的内容:要
11、搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的叫指针所指向的内存区,还有指针本身所占据的内存区。内存区。1指针的类型 语法的角度看,你只要把指针声明语句语法的角度看,你只要把指针声明语句里的指针名字去掉,剩下的部分就是这里的指针名字去掉,剩下的部分就是这个指针的类型。这是指针本身所具有的个指针的类型。这是指针本身所具有的类型。让我们看看例一中各个指针的类型。让我们看看例一中各个指针的类型:类型:(1)int*ptr;/指针的类型是指针的类型是int*(2)char*ptr;/
12、指针的类型是指针的类型是char*(3)int*ptr;/指针的类型是指针的类型是int*(4)int(*ptr)3;/指针的类型是指针的类型是int(*)3(5)int*(*ptr)4;/指针的类型是指针的类型是int*(*)42.指针所指向的类型。指针所指向的类型。当你通过指针来访问指针所指向的内存区时,当你通过指针来访问指针所指向的内存区时,指针所指向的类型决定了编译器将把那片内存指针所指向的类型决定了编译器将把那片内存区里的内容当做什么来看待。从语法上看,你区里的内容当做什么来看待。从语法上看,你只须把指针声明语句中的指针名字和名字左边只须把指针声明语句中的指针名字和名字左边的指针声明
13、符的指针声明符*去掉,剩下的就是指针所指向去掉,剩下的就是指针所指向的类型。例如:的类型。例如:(1)int*ptr;/指针所指向的类型是指针所指向的类型是int(2)char*ptr;/指针所指向的的类型是指针所指向的的类型是char(3)int*ptr;/指针所指向的的类型是指针所指向的的类型是int*(4)int(*ptr)3;/指针所指向的的类型是指针所指向的的类型是int()3(5)int*(*ptr)4;/指针所指向的的类型是指针所指向的的类型是int*()4 在指针的算术运算中,指针所指向的类型有很在指针的算术运算中,指针所指向的类型有很大的作用。大的作用。指针的类型指针的类型(
14、即指针本身的类型即指针本身的类型)和指针所指向和指针所指向的类型是两个概念。的类型是两个概念。3.指针的值,或者叫指针所指向的内存区或地址。指针的值,或者叫指针所指向的内存区或地址。指针的值是指针本身存储的数值,这个值指针的值是指针本身存储的数值,这个值将被编译器当作一个地址,而不是一个一将被编译器当作一个地址,而不是一个一般的数值。在般的数值。在32位程序里,所有类型的指位程序里,所有类型的指针的值都是一个针的值都是一个32位整数,因为位整数,因为32位程序位程序里内存地址全都是里内存地址全都是32位长。位长。指针所指向的内存区就是从指针的值所代指针所指向的内存区就是从指针的值所代表的那个内
15、存地址开始,长度为表的那个内存地址开始,长度为sizeof(指针指针所指向的类型所指向的类型)的一片内存区。的一片内存区。以后,我们说一个指针的值是以后,我们说一个指针的值是XX,就相当,就相当于说该指针指向了以于说该指针指向了以XX为首地址的一片内为首地址的一片内存区域;我们说一个指针指向了某块内存存区域;我们说一个指针指向了某块内存区域,就相当于说该指针的值是这块内存区域,就相当于说该指针的值是这块内存区域的首地址。区域的首地址。指针所指向的内存区和指针所指向的类指针所指向的内存区和指针所指向的类型是两个完全不同的概念。在例一中,型是两个完全不同的概念。在例一中,指针所指向的类型已经有了,
16、但由于指指针所指向的类型已经有了,但由于指针还未初始化,所以它所指向的内存区针还未初始化,所以它所指向的内存区是不存在的,或者说是无意义的。是不存在的,或者说是无意义的。以后,每遇到一个指针,都应该问问:以后,每遇到一个指针,都应该问问:这个指针的类型是什么?指针指向的类这个指针的类型是什么?指针指向的类型是什么?该指针指向了哪里?型是什么?该指针指向了哪里?4.指针本身所占据的内存区 指针本身占了多大的内存?你只要用函指针本身占了多大的内存?你只要用函数数sizeof(指针的类型指针的类型)测一下就知道了。测一下就知道了。在在32位平台里,指针本身占据了位平台里,指针本身占据了4个字节个字节
17、的长度。的长度。指针本身占据的内存这个概念在判断一指针本身占据的内存这个概念在判断一个指针表达式是否是左值时很有用。个指针表达式是否是左值时很有用。(二)指针的算术运算 指针可以加上或减去一个整数。指针的指针可以加上或减去一个整数。指针的这种运算的意义和通常的数值的加减这种运算的意义和通常的数值的加减运算的意义是不一样的。运算的意义是不一样的。例例1:1.chara20;2.int*ptr=a;.3.ptr+;例例2:intarray20;int*ptr=array;./此处略去为整型数组赋值的代码。此处略去为整型数组赋值的代码。.for(i=0;ia;ptr-b;ptr-c;(七)指针和函数
18、的关系(七)指针和函数的关系 可以把一个指针声明成为一个指向函数可以把一个指针声明成为一个指向函数的指针。的指针。intfun1(char*,int);int(*pfun1)(char*,int);pfun1=fun1;.inta=(*pfun1)(abcdefg,7);/通过函数通过函数指针调用函数。指针调用函数。可以把指针作为函数的形参。在函数调可以把指针作为函数的形参。在函数调用语句中,可以用指针表达式来作为实用语句中,可以用指针表达式来作为实参。参。例:例:int fun(char*);int a;char str=abcdefghijklmn;a=fun(str);.int fun(
19、char*s)int num=0;for(int i=0;i num+=*s;s+;return num;这个例子中的函数这个例子中的函数fun统计一个字符串中各统计一个字符串中各个字符的个字符的ASCII码值之和。前面说了,数组码值之和。前面说了,数组的名字也是一个指针。在函数调用中,当的名字也是一个指针。在函数调用中,当把把str作为实参传递给形参作为实参传递给形参s后后,实际是把,实际是把str的值传递给了的值传递给了s,s所指向的所指向的地址就和地址就和str所指向的地址一致,但是所指向的地址一致,但是str和和s各自占用各自的存储空间。在函数体内对各自占用各自的存储空间。在函数体内对
20、s进行自加进行自加1运算,并不意味着同时对运算,并不意味着同时对str进行进行了自加了自加1运算。运算。(八)指针类型转换(八)指针类型转换当我们初始化一个指针或给一个指针赋值时,赋值号的左边是一当我们初始化一个指针或给一个指针赋值时,赋值号的左边是一个指针,赋值号的右边是一个指针表达式。在我们前面所举的例个指针,赋值号的右边是一个指针表达式。在我们前面所举的例子中,绝大多数情况下,指针的类型和指针表达式的类型是一样子中,绝大多数情况下,指针的类型和指针表达式的类型是一样的,指针所指向的类型和指针表达式所指向的类型是一样的。的,指针所指向的类型和指针表达式所指向的类型是一样的。例:例:floa
21、tf=12.3;float*fptr=&f;int*p;在上面的例子中,假如我们想让指针在上面的例子中,假如我们想让指针p指向实数指向实数f,应该怎么搞?,应该怎么搞?是用下面的语句吗?是用下面的语句吗?p=&f;不对。因为指针不对。因为指针p的类型是的类型是int*,它指向的类型是,它指向的类型是int。表达式。表达式&f的结果的结果是一个指针,指针的类型是是一个指针,指针的类型是float*,它指向的类型是它指向的类型是float。两者不一致,。两者不一致,直接赋值的方法是不行的。至少在我的直接赋值的方法是不行的。至少在我的MSVC+6.0上,对指针的赋值上,对指针的赋值语句要求赋值号两边
22、的类型一致,所指向的类型也一致,其它的编译语句要求赋值号两边的类型一致,所指向的类型也一致,其它的编译器上我没试过,大家可以试试。为了实现我们的目的,需要进行器上我没试过,大家可以试试。为了实现我们的目的,需要进行“强强制类型转换制类型转换”:p=(int*)&f;如果有一个指针如果有一个指针p,我们需要把它的类型和所指向的类型改为,我们需要把它的类型和所指向的类型改为TYEP*和和TYPE,那么语法格式是:,那么语法格式是:(TYPE*)p;这样强制类型转换的结果是一个新指针,该新指针的类型是这样强制类型转换的结果是一个新指针,该新指针的类型是TYPE*,它指向的类型是它指向的类型是TYPE
23、,它指向的地址就是原指针指向的地址。而原来,它指向的地址就是原指针指向的地址。而原来的指针的指针p的一切属性都没有被修改。的一切属性都没有被修改。(九)指针的安全问题(九)指针的安全问题 例:例:chars=a;int*ptr;ptr=(int*)&s;*ptr=1298;指针指针ptr是一个是一个int*类型的指针,它指向类型的指针,它指向的类型是的类型是int。它指向的地址就是。它指向的地址就是s的首地的首地址。在址。在32位程序中,位程序中,s占一个字节,占一个字节,int类类型占四个字节。最后一条语句不但改变型占四个字节。最后一条语句不但改变了了s所占的一个字节,还把和所占的一个字节,
24、还把和s相临的高相临的高地址方向的三个字节也改变了。这三个地址方向的三个字节也改变了。这三个字节是干什么的?只有编译程序知道,字节是干什么的?只有编译程序知道,而写程序的人是不太可能知道的。也许而写程序的人是不太可能知道的。也许这三个字节里存储了非常重要的数据,这三个字节里存储了非常重要的数据,也许这三个字节里正好是程序的一条代也许这三个字节里正好是程序的一条代码,而由于你对指针的马虎应用,这三码,而由于你对指针的马虎应用,这三个字节的值被改变了!这会造成崩溃性个字节的值被改变了!这会造成崩溃性的错误。的错误。在用指针访问数组的时候,也要注意不在用指针访问数组的时候,也要注意不要超出数组的低端
25、和高端界限,否则要超出数组的低端和高端界限,否则也会造成类似的错误。也会造成类似的错误。在指针的强制类型转换:在指针的强制类型转换:ptr1=(TYPE*)ptr2中,如果中,如果sizeof(ptr2的的类型类型)大大于于sizeof(ptr1的类型的类型),那么在使用指针,那么在使用指针ptr1来访问来访问ptr2所指向的存储区时是安所指向的存储区时是安全的。如果全的。如果sizeof(ptr2的类型的类型)小于小于sizeof(ptr1的类型的类型),那么在使用指针,那么在使用指针ptr1来访问来访问ptr2所指向的存储区时是不安全所指向的存储区时是不安全的。至于为什么,读者结合例十七来
26、想的。至于为什么,读者结合例十七来想一一想,应该会明白的。想,应该会明白的。例:例:chara;int*ptr=&a;.ptr+;*ptr=115;该例子完全可以通过编译,并能执行。但是看到没有?第该例子完全可以通过编译,并能执行。但是看到没有?第3句对句对指针指针ptr进行自加进行自加1运算后,运算后,ptr指向了和整形变量指向了和整形变量a相邻的高地址相邻的高地址方向的一块存储区。这块存储区里是什么?我们不知道。有可能方向的一块存储区。这块存储区里是什么?我们不知道。有可能它是一个非常重要的数据,甚至可能是一条代码。而第它是一个非常重要的数据,甚至可能是一条代码。而第4句竟然句竟然往这片存储区里写入一个数据!这是严重的错误。所以在使用指往这片存储区里写入一个数据!这是严重的错误。所以在使用指针时,程序员心里必须非常清楚:我的指针究竟指向了哪里。针时,程序员心里必须非常清楚:我的指针究竟指向了哪里。