本节主要介绍Turbo C2.0标准输入输出函数和文件的输入输出函数。通过本 节的学习可以使读者掌握Turbo C2.0的屏幕输出、键盘输入输出以及磁盘文件的 读写函数, 并能开始进行一些简单的程序的编写。 控制流程语句主要包括: 条件语句、循环语句和开关语句。
1.1 标准输入输出函数 1.1.1 格式化输入输出函数 Turbo C2.0 标准库提供了两个控制台格式化输入、 输出函数printf( ) 和 scanf(), 这两个函数可以在标准输入输出设备上以各种不同的格式读写数据。 printf()函数用来向标准输出设备(屏幕)写数据; scanf() 函数用来从标准输入 设备(键盘)上读数据。下面详细介绍这两个函数的用法。 一、printf()函数 printf()函数是格式化输出函数, 一般用于向标准输出设备按规定格式输出 信息。在编写程序时经常会用到此函数。printf()函数的调用格式为: printf("<格式化字符串>", <参量表>); 其中格式化字符串包括两部分内容: 一部分是正常字符, 这些字符将按原 样输出; 另一部分是格式化规定字符, 以"%"开始, 后跟一个或几个规定字符, 用来确定输出内容格式。 参量表是需要输出的一系列参数, 其个数必须与格式化字符串所说明的输出 参数个数一样多, 各参数之间用","分开, 且顺序一一对应, 否则将会出现意想 不到的错误。
1. 格式化规定符 Turbo C2.0提供的格式化规定符如下: ━━━━━━━━━━━━━━━━━━━━━━━━━━ 符号 作用 ────────────────────────── %d 十进制有符号整数 %u 十进制无符号整数 %f 浮点数 %s 字符串 %c 单个字符 %p 指针的值 %e 指数形式的浮点数 %x, %X 无符号以十六进制表示的整数 %0 无符号以八进制表示的整数 %g 自动选择合适的表示法 ━━━━━━━━━━━━━━━━━━━━━━━━━━ 说明: (1). 可以在"%"和字母之间插进数字表示最大场宽。 例如: %3d 表示输出3位整型数, 不够3位右对齐。 %9.2f 表示输出场宽为9的浮点数, 其中小数位为2, 整数位为6, 小数点占一位, 不够9位右对齐。 %8s 表示输出8个字符的字符串, 不够8个字符右对齐。 如果字符串的长度、或整型数位数超过说明的场宽, 将按其实际长度输出。 但对浮点数, 若整数部分位数超过了说明的整数位宽度, 将按实际整数位输出; 若小数部分位数超过了说明的小数位宽度, 则按说明的宽度以四舍五入输出。 另外, 若想在输出值前加一些0, 就应在场宽项前加个0。 例如: %04d 表示在输出一个小于4位的数值时, 将在前面补0使其总宽度 为4位。 如果用浮点数表示字符或整型量的输出格式, 小数点后的数字代表最大宽度, 小数点前的数字代表最小宽度。 例如: %6.9s 表示显示一个长度不小于6且不大于9的字符串。若大于9, 则 第9个字符以后的内容将被删除。 (2). 可以在"%"和字母之间加小写字母l, 表示输出的是长型数。 例如: %ld 表示输出long整数 %lf 表示输出double浮点数 (3). 可以控制输出左对齐或右对齐, 即在"%"和字母之间加入一个"-" 号可 说明输出为左对齐, 否则为右对齐。 例如: %-7d 表示输出7位整数左对齐 %-10s 表示输出10个字符左对齐 2. 一些特殊规定字符 ━━━━━━━━━━━━━━━━━━━━━━━━━━ 字符 作用 ────────────────────────── /n 换行 /f 清屏并换页 /r 回车 /t Tab符 /xhh 表示一个ASCII码用16进表示, 其中hh是1到2个16进制数 ━━━━━━━━━━━━━━━━━━━━━━━━━━ 由本节所学的printf()函数, 并结合上一节学习的数据类型, 编制下面的程 序, 以加深对Turbo C2.0数据类型的了解。 例1 #include<stdio.h> #include<string.h> int main() { char c, s[20], *p; int a=1234, *i; float f=3.141592653589; double x=0.12345678987654321; p="How do you do"; strcpy(s, "Hello, Comrade"); *i=12; c='/x41'; printf("a=%d/n", a); /*结果输出十进制整数a=1234*/ printf("a=%6d/n", a); /*结果输出6位十进制数a= 1234*/ printf("a=%06d/n", a); /*结果输出6位十进制数a=001234*/ printf("a=%2d/n", a); /*a超过2位, 按实际值输出a=1234*/ printf("*i=%4d/n", *i); /*输出4位十进制整数*i= 12*/ printf("*i=%-4d/n", *i); /*输出左对齐4位十进制整数*i=12*/ printf("i=%p/n", i); /*输出地址i=06E4*/ printf("f=%f/n", f); /*输出浮点数f=3.141593*/ printf("f=6.4f/n", f); /*输出6位其中小数点后4位的浮点数 f=3.1416*/ printf("x=%lf/n", x); /*输出长浮点数x=0.123457*/ printf("x=%18.16lf/n", x);/*输出18位其中小数点后16位的长浮点 数x=0.1234567898765432*/ printf("c=%c/n", c); /*输出字符c=A*/ printf("c=%x/n", c); /*输出字符的ASCII码值c=41*/ printf("s[]=%s/n", s); /*输出数组字符串s[]=Hello, Comrade*/ printf("s[]=%6.9s/n", s);/*输出最多9个字符的字符串s[]=Hello, Co*/ printf("s=%p/n", s); /*输出数组字符串首字符地址s=FFBE*/ printf("*p=%s/n", p); /* 输出指针字符串p=How do you do*/ printf("p=%p/n", p); /*输出指针的值p=0194*/ getch(); retunr 0; } 上面结果中的地址值在不同计算机上可能不同。 例1.中第一条语句#include<stdio.h>的含义是调用另一个文件stdio.h, 这 是一个头文件, 其中包括全部标准输入输出库函数的数据类型定义和函数说明。 Turbo C2.0对每个库函数便用的变量及函数类型都已作了定义与说明, 放在相应 头文件"*.h"中, 用户用到这些函数时必须要用#include<*.h>或#include"*. h" 语句调用相应的头文件, 以供连接。若没有用此语句说明, 则连接时将会出现错 误。 二、scanf()函数 scanf()函数是格式化输入函数, 它从标准输入设备(键盘) 读取输入的信息。 其调用格式为: scanf("<格式化字符串>", <地址表>); 格式化字符串包括以下三类不同的字符; 1. 格式化说明符: 格式化说明符与printf()函数中的格式说明符基本相同。 2. 空白字符: 空白字符会使scanf()函数在读操作中略去输入中的一个或多 个空白字符。 3. 非空白字符: 一个非空白字符会使scanf()函数在读入时剔除掉与这个非 空白字符相同的字符。 地址表是需要读入的所有变量的地址, 而不是变量本身。这与printf()函数 完全不同, 要特别注意。各个变量的地址之间同","分开。 例2: main() { int i, j; printf("i, j=?/n"); scanf("%d, %d", &i, &j); }
上例中的scanf()函数先读一个整型数, 然后把接着输入的逗号剔除掉, 最 后读入另一个整型数。如果","这一特定字符没有找到, scanf()函数就终止。若 参数之间的分隔符为空格, 则参数之间必须输入一个或多个空格。 说明: (1). 对于字符串数组或字符串指针变量, 由于数组名和指针变量名本身就 是地址, 因此使用scanf()函数时, 不需要在它们前面加上"&"操作符。 例3 mian() { char *p, str[20]; scanf("%s", p); /*从健盘输入字符串*/ scanf("%s", str); printf("%s/n", p); /*向屏幕输出字符串*/ printf("%s/n", str); }
(2). 可以在格式化字符串中的"%"各格式化规定符之间加入一个整数, 表示 任何读操作中的最大位数。 如例3中若规定只能输入10字符给字符串指针p, 则第一条scanf() 函数语句 变为 scanf("%10s", p); 程序运行时一旦输入字符个数大于10, p就不再继续读入, 而后面的一个读 入函数即scanf("%s", str)就会从第11个字符开始读入。 实际使用scanf()函数时存在一个问题, 下面举例进行说明: 当使用多个scanf()函数连续给多个字符变量输入时, 例如: main() { char c1, c2; scanf("%c", &c1); scanf("%c", &c2); printf("c1 is %c, c2 is %c", c2/1, c2); }
运行该程序, 输入一个字符A后回车 (要完成输入必须回车), 在执行scanf ("%c", &c1)时, 给变量c1赋值"A", 但回车符仍然留在缓冲区内, 执行输入语句 scanf("%c", &c2)时, 变量c2输出的是一空行, 如果输入AB后回车, 那么输出结 果为: c1 is A, c2 is B。 要解决以上问题, 可以在输入函数前加入清除函数fflush()( 这个函数的使 用方法将在本节最后讲述)。修改以上程序变成: #include<stdio.h> main() { char c1, c2; scanf("%c", &c1); fflush(stdin); scanf("%c", &c2); printf("c1 is %c, c2 is %c", c1, c2); }
1.1.2 非格式化输入输出函数 非格式化输入输出函数可以由上面讲述的标准格式化输入输出函数代替, 但 这些函数编译后代码少, 相对占用内存也小, 从而提高了速度, 同时使用也比较 方便。下面分别进行介绍。 一、puts()和gets()函数 1. puts()函数 puts()函数用来向标准输出设备(屏幕)写字符串并换行, 其调用格式为: puts(s); 其中s为字符串变量(字符串数组名或字符串指针)。 puts()函数的作用与语printf("%s/n", s)相同。 例4: main() { char s[20], *f; /*定义字符串数组和指针变量*/ strcpy(s, "Hello! Turbo C2.0"); /*字符串数组变量赋值*/ f="Thank you"; /*字符串指针变量赋值*/ puts(s); puts(f); }
说明: (1). puts()函数只能输出字符串, 不能输出数值或进行格式变换。 (2). 可以将字符串直接写入puts()函数中。如: puts("Hello, Turbo C2.0");
2. gets()函数 gets()函数用来从标准输入设备(键盘)读取字符串直到回车结束, 但回车符 不属于这个字符串。其调用格式为: gets(s); 其中s为字符串变量(字符串数组名或字符串指针)。 gets(s)函数与scanf("%s", &s)相似, 但不完全相同, 使用scanf("%s", &s) 函数输入字符串时存在一个问题, 就是如果输入了空格会认为输入字符串结束, 空格后的字符将作为下一个输入项处理, 但gets() 函数将接收输入的整个字符 串直到回车为止。 例5 main() { char s[20], *f; printf("What's your name?/n"); gets(s); /*等待输入字符串直到回车结束*/ puts(s); /*将输入的字符串输出*/ puts("How old are you?"); gets(f); puts(f); } 说明: (1). gets(s)函数中的变量s为一字符串。如果为单个字符, 编译连接不会 有错误, 但运行后会出现"Null pointer asignmemt"的错误。
二、putchar()、getch()、getche()和getchar()函数 1. putchar()函数 putchar()函数是向标准输出设备输出一个字符, 其调用格式为: putchar(ch); 其中ch为一个字符变量或常量。 putchar()函数的作用等同于printf("%c", ch); 例6: #include<stdio.h> main() { char c: /*定义字符变量*/ c='B'; /*给字符变量赋值*/ putchar(c); /*输出该字符*/ putchar('/x42'); /*输出字母B*/ putchar(0x42); /*直接用ASCII码值输出字母B*/ } 从本例中的连续四个字符输出函数语句可以分清字符变量的不同赋值方法。 2. getch()、getche()和getchar()函数 (1) getch()和getche()函数 这两个函数都是从键盘上读入一个字符。其调用格式为: getch(); getche(); 两者的区别是: getch()函数不将读入的字符回显在显示屏幕上, 而getche() 函数却将读入的字符回显到显示屏幕上。 例7: #include<stdio.h> main() { char c, ch; c=getch(); /*从键盘上读入一个字符不回显送给字符变量c*/ putchar(c); /*输出该字符*/ ch=getche(); /*从键盘上带回显的读入一个字符送给字符变量ch*/ putchar(ch); } 利用回显和不回显的特点, 这两个函数经常用于交互输入的过程中完成暂停 等功能。 例8: #include<stdio.h> main() { char c, s[20]; printf("Name:"); gets(s); printf("Press any key to confinue..."); getch(); /*等待输入任一键*/ } (2) getchar()函数 getchar()函数也是从键盘上读入一个字符, 并带回显。它与前面两个函数 的区别在于: getchar()函数等待输入直到按回车才结束, 回车前的所有输入字 符都会逐个显示在屏幕上。但只有第一个字符作为函数的返回值。 getchar()函数的调用格式为: getchar(); 例9: #include<stdio.h> main() { char c; c=getchar(); /*从键盘读入字符直到回车结束*/ putchar(c); /*显示输入的第一个字符*/ getch(); /*等待按任一健*/ } 2. fclose()函数 fclose()函数用来关闭一个由fopen()函数打开的文件 , 其调用格式为: int fclose(FILE *stream); 该函数返回一个整型数。当文件关闭成功时, 返回0, 否则返回一个非零值。 可以根据函数的返回值判断文件是否关闭成功。 例10: #iclude<stdio.h> main() { FILE *fp; /*定义一个文件指针*/ int i; fp=fopen("CLIB", "rb"); /*打开当前目录名为CLIB的文件只读*/ if(fp==NULL) /*判断文件是否打开成功*/ puts("File open error");/*提示打开不成功*/ i=fclose(fp); /*关闭打开的文件*/ if(i==0) /*判断文件是否关闭成功*/ printf("O,K"); /*提示关闭成功*/ else puts("File close error");/*提示关闭不成功*/ }
二、有关文件操作的函数 本节所讲的文件读写函数均是指顺序读写, 即读写了一条信息后, 指针自动 加1。下面分别介绍写操作函数和读操作函数。
1. 文件的顺序写函数 fprintf()、fputs()和fputc()函数 函数fprintf()、fputs()和fputc()均为文件的顺序写操作函数, 其调用格 式如下: int fprintf(FILE *stream, char *format, <variable-list>); int fputs(char *string, FILE *steam); int fputc(int ch, FILE *steam); 上述三个函数的返回值均为整型量。fprintf() 函数的返回值为实际写入文 件中的字罕个数(字节数)。如果写错误, 则返回一个负数, fputs()函数返回0时 表明将string指针所指的字符串写入文件中的操作成功, 返回非0时, 表明写操 作失败。fputc()函数返回一个向文件所写字符的值, 此时写操作成功, 否则返 回EOF(文件结束结束其值为-1, 在stdio.h中定义)表示写操作错误。 fprintf( ) 函数中格式化的规定与printf( ) 函数相同, 所不同的只是 fprintf()函数是向文件中写入。而printf()是向屏幕输出。 下面介绍一个例子, 运行后产后一个test.dat的文件。 例11: #include<stdio.h> main() { char *s="That's good news"); /*定义字符串指针并初始化*/ int i=617; /*定义整型变量并初始化*/ FILE *fp; /*定义文件指针*/ fp=fopne("test.dat", "w"); /*建立一个文字文件只写*/ fputs("Your score of TOEFLis", fp);/*向所建文件写入一串字符*/ fputc(':', fp); /*向所建文件写冒号:*/ fprintf(fp, "%d/n", i); /*向所建文件写一整型数*/ fprintf(fp, "%s", s); /*向所建文件写一字符串*/ fclose(fp); /*关闭文件*/ } 用DOS的TYPE命令显示TEST.DAT的内容如下所示: 屏幕显示 Your score of TOEFL is: 617 That's good news
2. 文件的顺序读操作函数 fscanf()、fgets()和fgetc()函数 函数fscanf()、fgets()和fgetc()均为文件的顺序读操作函数, 其调用格式 如下: int fscanf(FILE *stream, char *format, <address-list>); char fgets(char *string, int n, FILE *steam); int fgetc(FILE *steam); fscanf()函数的用法与scanf()函数相似, 只是它是从文件中读到信息。 fscanf()函数的返回值为EOF(即-1), 表明读错误, 否则读数据成功。fgets()函 数从文件中读取至多n-1个字符(n用来指定字符数), 并把它们放入string指向的 字符串中, 在读入之后自动向字符串未尾加一个空字符, 读成功返回string指针, 失败返回一个空指针。fgetc()函数返回文件当前位置的一个字符, 读错误时返 回EOF。 下面程序读取例11产生的test.dat文件, 并将读出的结果显示在屏幕上。 例12 #include<stdio.h> main() { char *s, m[20]; int i; FILE *fp; fp=fopen("test.dat", "r"); /*打开文字文件只读*/ fgets(s, 24, fp); /*从文件中读取23个字符*/ printf("%s", s); /*输出所读的字符串*/ fscanf(fp, "%d", &i); /*读取整型数*/ printf("%d", i); /*输出所读整型数*/ putchar(fgetc(fp)); /*读取一个字符同时输出*/ fgets(m, 17, fp); /*读取16个字符*/ puts(m); /*输出所读字符串*/ fclose(fp); /*关闭文件*/ getch(); /*等待任一键*/ } 运行后屏幕显示: Your score of TOEFL is: 617 That's good news 如果将上例中fscanf(fp, "%d", &i)改为fscanf(fp, "%s", m), 再将其后 的输出语句改为printf("%s", m), 则可得出同样的结果。由此可见Turbo C2. 0 中只要是读文字文件, 则不论是字符还是数字都将按其ASCII值处理。 另外还要 说明的一点就是fscanf()函数读到空白符时, 便自动结束, 在使用时要特别注意。 3. 文件的随机读写 有时用户想直接读取文件中间某处的信息, 若用文件的顺序读写必须从文件 头开始直到要求的文件位置再读, 这显然不方便。Turbo C2.0提供了一组文件的 随机读写函数, 即可以将文件位置指针定位在所要求读写的地方直接读写。 文件的随机读写函数如下: int fseek (FILE *stream, long offset, int fromwhere); int fread(void *buf, int size, int count, FILE *stream); int fwrite(void *buf, int size, int count, FILE *stream); long ftell(FILE *stream); fseek()函数的作用是将文件的位置指针设置到从fromwhere开始的第offset 字节的位置上, 其中fromwhere是下列几个宏定义之一: 文件位置指针起始计算位置fromwhere ━━━━━━━━━━━━━━━━━━━━━━━━━━━ 符号常数 数值 含义 ─────────────────────────── SEEK_SET 0 从文件开头 SEEK_CUR 1 从文件指针的现行位置 SEEK_END 2 从文件末尾 ━━━━━━━━━━━━━━━━━━━━━━━━━━━ offset是指文件位置指针从指定开始位置(fromwhere指出的位置)跳过的字 节数。它是一个长整型量, 以支持大于64K字节的文件。fseek()函数一般用于对 二进制文件进行操作。 当fseek()函数返回0时表明操作成功, 返回非0表示失败。 下面程序从二进制文件test_b.dat中读取第8个字节。 例13: #include<stdio.h> main() { FILE *fp; if((fp=fopen("test_b.dat", "rb"))==NULL) { printf("Can't open file"); exit(1); } fseek(fp, 8. 1, SEEK_SET); fgetc(fp); fclose(fp); } fread()函数是从文件中读count个字段, 每个字段长度为size个字节, 并把 它们存放到buf指针所指的缓冲器中。 fwrite()函数是把buf指针所指的缓冲器中, 长度为size个字节的count个字 段写到stream指向的文件中去。 随着读和写字节数的增大, 文件位置指示器也增大, 读多少个字节, 文件位 置指示器相应也跳过多少个字节。读写完毕函数返回所读和所写的字段个数。 ftell()函数返回文件位置指示器的当前值, 这个值是指示器从文件头开始 算起的字节数, 返回的数为长整型数, 当返回-1时, 表明出现错误。 |