欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

《C primer plus》笔记:文件

程序员文章站 2023-12-22 21:31:34
...

20180607 qzd


1. 文件通信


  • 重定向:books > bklist 可能会把不必要的东西输出到文件
  • 文件:一系列连续的字节,文本模式,二进制模式
  • 文件模式:所有文件内容都以二进制形式存储
  • 文本文件:最初使用二进制编码的字符表示文本
  • 二进制文件:二进制值代表机器语言代码或数值数据或图片或音乐编码
  • 不同操作系统处理不同 MS-DOS读文件\r\n–>\n写文件\n–>\r\n
  • I/O级别:底层I/O使用操作系统标准I/O,标准高级I/O使用stdio.h头文件的定义的(通用)
  • 标准文件:C打开3个文件,标准输入(键盘),标准输出,标准错误输出(显示器)

2. 标准I/O


  • 好处:可移植性强,专门函数简化了操作,输入输出优势缓冲的
  • 演示:
#include <stdio.h>
#include <stdlib.h>
int main(int argc, char * argv[]){
 int ch;       //读取文件,存储每个字符的地方
 FILE *fp;     //文件指针
 unsigned long count = 0;
 if(argc != 2){
 printf("Usage: %s filename \n", argv[0]);
 }
 if( (fp = fopen(argv[1], "r")) == NULL ){
 printf("can't open %s\n", argv[1]);
 exit(EXIT_FAILURE);
 }
 while( (ch = getc(fp)) != EOF ){
 putc(ch, stdout);
 count++;
 }
 fclose(fp);
 printf("File %s has %lu characters\n", argv[1], count);
 return 0;

}
  • 检查命令行参数:exit()关闭文件并结束程序(0或EXIT_SUCCESS结束成功,EXIT _FALLURE结束失败)
  • fopen函数:打开文件,成功打开后返回文件指针(FILE*)
FILE * fp = fopen("words", "r");
模式 含义
r 读模式
w 写模式(文件长度截0,可以创建新文件)
a 写模式(添加字符,可以创建新文件)
r+ 更新模式(读写)
w+ 更新模式(读写,截断,创新)
a+ 更新模式(读写,添加,创新)
*b 二进制文件用法同上
  • getc和putc函数:与getchar,putchar类似
ch = getchar();   ch = getc(fp);   //文件输入
ch = putchar();   ch = putc(fpout) //文件输出
  • 文件结尾:避免读到空文件,使用入口循环
while( (ch = getc(fp)) != EOF) { putc(fpout); }
  • fclose函数:关闭指定的文件,必要刷新缓冲区
if (fclose(fp) != 0 )  //成功放回0,否则返回EOF
 printf("error in closing file %s\n", argv[1]);
  • 指向标准文件的指针
标准文件 文件指针 设备
标准输入 stdin 键盘
标准输出 stdou 显示器
标准错误 stder 显示器

3. 文件压缩程序


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define N 40
int main(int argc, char * argv[]){
 FILE *in, *out;
 int ch;
 char name[N];
 int count;
 //检查命令行参数
 if(argc < 2){
 fprintf(stderr, "Usage:%s filename\n", argv[0]);
 exit(EXIT_FAILURE);
 }
 //设置输入
 if( (in = fopen(argv[1], "r")) == NULL){
 fprintf(stderr, "I could not open this file\" %s \" \n ", argv[1]);
 exit(EXIT_FAILURE);
 }
 //设置输出
 strncpy(name, argv[1], N - 5);
 name[N -5] = '\0';
 strcat(name, ".red");
 if( (out = fopen(name, "w")) == NULL ){
 fprintf(stderr, "I could not open this file");
 exit(3);
 }
 //拷贝
 while( (ch = getc(in)) != EOF ){
 if (count++ % 3 == 0)
 putc(ch, out);
 }
 //关闭文件
 if( fclose(in) != 0 || fclose(out) != 0 )
 fprintf(stderr, "Error in closing files\n");
 return 0;
}

4. 文件I/O


  • fprinf和fscanf函数:与printf和scanf类似,区别在于,前面需要第一个参数待处理的文件。
fprintf(fp, "%s\n", words);
fscanf(fp, "%s\n", words);
  • fgets和fputs函数:
fgets(buf, LEN, fp);    // buf char型数组的名称,fp指针
fputs(buf, fp);

5. 随机访问


  • fseek() ftell()函数:文件看成数组,fseek返回int.ftell返回long

fseek(fp, 0L, SEEK_END); //定位到文件末尾
// 1)FILE指针,指向待查找的文件
// 2)偏移量,从起始点开始要移动的距离,正负0
// 3)模式,SEEK_SET文件开始, SEEK_CUR当前位置,SEEK_END文件末尾
// 正常返回0,c出现错误返回-1
ftell(fp) //返回当前位置

  • 二进制模式和文本模式:
    • UNIX只有一种文件格式,不需要特殊转换,MS-DOS很多编辑器都是用ctrl+z来标记文件结尾的二进制文件和文本文件模式的另一个不同的地方,
    • MS_DOS用户\r\n来标识文件结尾,c程序会把\r\n看做成\n,对于ftell()返回值把\r\n当作一个字节来计数
  • 可移植性:
    • 二进制模式中,实现不必支持SEEK_END模式,因此无法保证程序的可移植性,更好的办法是一个一个字节读取整个文件
    • 文本模式中,只有使用fseek才能保证其相应的行为
  • fgetpos函数,fsetpos函数:新增的定位函数,突破long的限制
int fgetpos(FILE * restrict stream, fpos_t * restrict pos); // 成功返回0,失败返回非0
// 把fpos_t的值放在pos指向的位置
int fsetpos(FILE * stream, const fpos_t * pos); //成功返回0,失败返回非0
// 使用fpos_t设置文件指针指向该值指向的位置

6. 标准IO机理


//第一步:调用fopen打开文件
//1)自动打开3个标准文件 
//2)打开文件 
//3)打开一个或者两个缓冲区以及其文件缓冲的数据结构 
//4)返回一个指向该结构的指针
FILE * fp = fopen("words", "a+");

//第二步:调用stdio.h中的输入函数
//1)文件的数据块被拷贝至缓冲区
//2)设置缓冲区大小,设置fp所指向结构中的值(流中当前位置和拷贝进缓冲区的字节数)

7. 其他标准IO函数


函数 作用
int ungetc(int, FILE*) c指定的字符放回输入流
int fflush(FILE*) 刷新缓冲区
int setcbuf(FILE*,char *, int, size_t) 创建一个供io替换的缓冲区
size_t fread() 二进制out
size_t fwirte() 二进制in
  • size_t fwrite函数:返回sizeof运算后的类型
size_t fwrite(const void * restrict ptr, size_t size, size_t nmemb, FILE * restrict fp);
// ptr 待写入的地址,size数据块的大小, nmemb带写入数据块的数量,fp待写入的文件
char buffer[256];
fwrite(buffer, 256, 1, fp);
double earnings[10];
fwrite(earings, sizeof(double), 10, fp);
  • size_t fread函数:返回sizeof运算后的类型
size_t fwrite(void * restrict ptr, size_t size, size_t nmemb, FILE * restrict fp);
// ptr 待读取文件数据在内存中的地址,size数据块的大小, nmemb带写入数据块的数量,fp待读取的文件
char buffer[256];
fwrite(buffer, 256, 1, fp);
double earnings[10];
fwrite(earings, sizeof(double), 10, fp);

参考:https://suoyuesmile.github.io/2015/07/10/c5/

上一篇:

下一篇: