linux 标准io笔记

三种缓冲

1.全缓冲:在缓冲区写满时输出到指定的输出端. 比如对磁盘上的文件进行读写通常是全缓冲的.

2.行缓冲:在遇到'\n'时输出到指定的输出端. 比如标准输入和标准输出就是行缓冲, 回车后就会进行相应的I/O操作.

3.无缓冲:有什么就输出什么. 比如标准错误输出, 出错时立即显示出来.

设置缓冲方式的函数有两个,分别为setbuf,setvbuf

可用fflush立即输出,不管缓冲区是否达到输出标准

打开、关闭

1.FILE *fopen(const char *restrict pathname, const char *restrict type)

第一个参数为文件的路径,

第二个参数为打开的方式: r/rb只读; w/wb只写; a/ab添加; r+/ra+读写; w+/wa+读写; a+/ab+文件尾添加读写

2.int fclose(FILE *fp)

关闭打开的文件流, 释放内存

读写终止判断

读写终止分两种情况, 读写到文件尾正常终止, 读写出错返回

int feof(FILE *fp): 如果是正常终止, 函数返回非0(true)值

int ferror(FILE *fp): 如果是出错终止, 函数返回非0(true)值

每次读写一个字符

getc/fgetc/getchar

putc/fputc/putchar

getchar和putchar对应终端窗口的输入输出, 不用指定参数

getc和fgetc需要指定FILE *fp

void onechar(FILE *fp){
int ch;
while((ch=getc(fp)) != EOF){
if(putchar(ch) == EOF){
printf("output error");
exit(1);
}
}
if(ferror(fp)){
printf("file read error");
exit(1);
}
}

每次读写一行

fgets/gets

fputs/puts

gets/puts默认对应终端窗口的输入输出

注意: puts会在行尾插入添加一个'\n', 所以当我们用puts打印文件时,每行显示时都有两个'\n'(除最后一行外), 也就是每行内容下面都会有一个空行

void  oneline(FILE *fp){
char buf[1024];
while(fgets(buf,MAXLINE,fp) != NULL){
if(puts(buf) == EOF){
printf("output error");
exit(1);
}
}
if(ferror(fp)){
printf("read error");
exit(1);
}
}

每次读写一个对象

这种读写方式也称为二进制方式, 由我们自己设定一次读几个字节

  • size_t fread(void *restrict ptr, size_t size, size_t nobj, FILE *restrict fp)

    参数含义:从fp读取, 读取的内容写到ptr指向的内存, 每次读取size个字符, 一共读nobj次, 返回值为实际读取的次数
  • size_t fwrite(const void *restrict ptr, size_t size, size_t nobj, FILE *restrict fp)
struct people{
char name[10];
int age;
}; void readstruct(){
struct people man;
FILE *fp;
if((fp=fopen("./test.txt","r")) == NULL){
printf("fopen error");
exit(1);
}
if(fread(&man,sizeof(struct people),1,fp) != 1){
printf("fread error");
exit(1);
}
printf("name=%s, age=%d\n",man.name,man.age);
fclose(fp);
} void writestruct(){
struct people man;
int i;
sprintf(man.name,"charlotte");
man.age=20;
FILE *fp;
if((fp=fopen("./test.txt","w+")) == NULL){
printf("fopen error");
exit(1);
}
if(fwrite(&man,sizeof(struct people),1,fp) != 1){
printf("fwrite error");
exit(1);
}
fclose(fp);
}

跳转

创建一个文件, 然后写入一些内容, 这时文件指针在文件尾,

如果想用这个文件指针读取刚刚写入的内容, 就需要先将文件指针移动文件头部

void rewind(FILE *fp)

将文件指针移动文件头

void fseek(FILE*fp, long offset, int whence)

随意移动到哪里

whence为初步移动: SEEK_SET 移到文件头; SEEK_END 移到文件尾; SEEK_CUR 移到当前位置(不动)

offset为精调:确定了whence之后, 再移动offset个位置

临时文件

临时文件的两个特殊需求:1.文件名要唯一 2.不需要时最好能自己删除

char *tmpnam(char *ptr):

产生唯一的"路径/文件名"形式的字符串,传入的参数可以为NULL,或者长度不小于L_tmpnam的字符数组

FILE *tmpfile()

自动生成文件名并创建临时文件, 临时文件在关闭或程序退出时自动删除

如果是一次性读写可以直接调用tmpfile, 如果需要重复读写可以用tmpnam生成文件名然后手动创建和删除文件

int main(){
printf("just test: %s\n",tmpnam(NULL)); char name[L_tmpnam];
tmpnam(name);
printf("name: %s\n",name); char buf[1024];
FILE *fp;
if((fp=tmpfile()) == NULL){
printf("tmpfile error");
exit(1);
}
fputs("one line\n",fp);
rewind(fp);
fgets(buf,1024,fp);
printf("%s",buf);
return 0;
}
上一篇:lua中打印所以类型功能实现table嵌套table


下一篇:javascript 数组和字符串的转化