一、I/O的概念
流
就C语言而言,所有的I/O操作就是简单的从程序移进、移出字节的事情,因此这种字节流被称为流。
绝大多数的流都是完全缓冲的。“读取”和“写入”都是在一块被称为:缓冲区的内存区被来回复制数据。从内存中来回复制数据是特别快的,用于输出流的缓冲区只有被写满的时候才会被刷新到设备或文件。因为一次性把放满的缓冲区数据写入和逐片把程序产生的数据写入相比较效率高。
显示器刷新方式为行缓冲。
流被分为两种:文本流和二进制流。
- 文本流在不同的系统中实现不太相同。
- 二进制流中的字节完全是按照程序编写他们形式写入到文件和设备,而且完全根据他们从文件或设备读取的形式读入到程序
文件
在stdio.h中定义了一个FILE结构。
FILE是一个数据结构,用于访问一个流。每一个流都对应一个FILE结构。
对于一个C程序而言,至少打开三个流:
- 标准输入(stdin)
- 标准输出(stdout)
- 标准错误(stderr)
标准输入通常为键盘设备,标准输出通常为终端或屏幕。
I/O常量
- EOF
文件结束标志,表示文件到了结尾 - FOPEN_MAX
一个程序最多打开文件数 - FILENAME_MAX
文件名的最大长度
二、关于流的操作
打开流
**FILE * fopen (
const char * filename, const char * mode);**
- filename:要打开的文件
- mode
- r :以只读方式打开
- w :以只写的方式打开
- a :以追加方式打开,如果文件不存在,则创建一个新文件
- r+ :以读写方式打开,文件必须存在
- w+: 以读写方式打开,如果文件不存在,则创建一个新文件
- a+:以读和追加方式打开,如果文件不存,则创建一个新文件
关闭流
int fclose (FILE*stream);
例:
#include <stdio.h>
int main(void)
{
FILE *fp;
fp = fopen("myfile.txt", "w");
if (NULL != fp)
{
fputs("fopen example", fp);
fclose(fp);
}
}
关于文件I/O的一些概况:
- 程序必须为同时处于活动状态的每个⽂件声明⼀个指针变量
向⼀个结构,当它处于活动状态时由流使⽤。 - 流通过fopen函数打开,打开流的时候,必须指定需要访问的
fopen函数和操作系统去验证⽂件或者设备确实存在,验证访问 - 根据需要对⽂件进⾏读取和写⼊。
- 最后, fclose函数可以关闭流。关闭⼀个流可以防⽌与它相关
储于缓冲区中的数据被正确的写⼊⽂件中。
三、IO函数
I/O函数有三种方式:单个字符、文本行、二进制数据
功能 | 函数名 | 适用于 |
---|---|---|
字符输入函数 | getchar | 标准输入流 |
字符输出函数 | putchar | 标准输出流 |
字符输入函数 | fgetc,getc | 所有输入流 |
字符输出函数 | fputc,putc | 所有输出流 |
文本行输入函数 | fgets,gets | 所有输入流 |
文本行输出函数 | fputs,puts | 所有输出流 |
格式化输入函数 | scanf | 标准输入流 |
格式化输出函数 | printf | 标准输出流 |
格式化输入函数 | fscanf | 所有输入流 |
格式化输出函数 | fprintf | 所有输出流 |
二进制输入 | fread | 文件 |
二进制输出 | fwrite | 文件按 |
简单mycpy
利用二进制I/O实现一个简单的mycpy;
size_t fread(void *ptr, size_t size,size_t count,FILE *stream);
size_t fwrite(const void*ptr, size_t size,size_t count,FILE* stream);
- ptr:是一个指针,对fread来说,它是读入数的存放地址,对fwrite来说,他是输出数据的地址
- size:要读写的字节数
- count:要进行读写多少个字节的数据项
- fp:文件型指针
#include <stdio.h>
#include <stdlib.h>
#include <stdio.h>
{
if(argc !=3)
{
printf("Usage: %s src dst\n", argv[0]);
return 1;
}
FILE *fin =NULL;
FILE *fout = NULL;
char *mem = NULL;
int ret = 0;
fin = fopen(argv[1], "rb");
if(NULL == fin)
{
perror("fopen");
ret = 1;
goto end;
}
fout = fopen(argv[2], "wb");
if(NULL ==fout)
{
perror("fopen");
ret = 1;;
goto end;
}
fseek(fin, 0, SEEK_END);
long lsize = ftell(fin); //获取文件指针的当前位置
rewind(fin);
mem = (char*)malloc(lsize);
if(NULL == mem)
{
perror("malloc");
ret = 3;
goto end;
}
fread(mem, lsize, 1,fin);
fwrite(mem , lsize, 1, fout);
end :
if(NULL != fin)
fclose(fin);
if(NULL != fout)
fclose(fin);
if(NULL != mem)
free(mem);
return ret;
}