博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
linux 标准I/O (二)
阅读量:5911 次
发布时间:2019-06-19

本文共 3014 字,大约阅读时间需要 10 分钟。

<Uinx 环境高级编程笔记>
 
以前经常遇到两种I/O操作
一类是f打头的fopen, fread, fwrite
一类是没有f打头的open, read, fwrite
原来一个是UNIX I/O(在类Unix系统上实现),另一类是标准I/O(在很多系统上都有实现,包括windows)
 

使用标准I/O的好处

因为在出UNIX的很多操作系统(包括Windows, linux)上都实现了此库,所有有利于软件的移植。
 
与UnixI/O的主要区别
UNIX I/O的函数都是针对文件描述符的,而标准I/O的操作都是围绕流进行的。所为的流就是一个FILE *
标准I/O提供了缓存--为了尽量减少write, read的调用。
标准I/O的效率会低一些,因为它是多了一层封装。即,fread通过调用read来实现。
 
标准I/O的头文件
stdio.h
三个标准I/O流预定义指针:stdin, stdout, stderr
(Unix I/O: STDIN_FILENO, STDOUT_FILENO, STDERR_FILENO)
 
缓存
有缓存就会有延迟,即输出设备上的内容和缓存中的内容很可能不一样。可以调用fflush刷新缓存。
有多种缓存类型,可以调用下面的API来更改默认缓存类型。
setbuf,  setvbuf
setvbuf可以精确的说明缓存的类型。
fclose关闭流时也会刷新流。
当一个进程
正常终止时(直接调用exit,或从main函数返回),则所有带未写缓存数据的标准I/O流都会被刷新,所有打开的标准I/O流都会被关闭。
 
标准I/O API
打开流
FILE * fopen(const char * path,const char * mode);
mode参数很简单:
r    文件只读
r+  文件可读写,文件必须存在
w   文件只写。相当于删除原文件,创建一个新文件。即若文件存在,长度变为0。若文件不存在,则创建。
w+ 文件可读写。其它和w相同。
a    以附加方式打开只写文件。文件不存在,会自动创建。
a+ 以附加方式打开读写文件。文件不存在,会自动创建。
b    以二进制方式操作文件。可以和上面的任意一个组合。
自动创建的文件访问权限位:644
读写流
每次一个字符的I/O
getc, fgetc, getchar
getc一般是宏调用,效率高于fgetc。fgetc可以作为一个地址传递给其它函数。
int getc(FILE * stream);
int fgetc(FILE *stream);
int getchar(void) 相当于 getc(stdin);
从一个流读取一个字符后,可以调用ungetc将字符再送回流中。
上面三个API出错或到达文件尾都返回-1。具体是什么情况,需要调用下面的两个API来判断:
int ferror(FILE *stream);
int feof(FILE *stream);
什么情况下会用到回送字符呢?
当正在读一个输入流,并进行某种形式的分字或分记号操作时,会经常用到回送字符操作。
有时需要先看一看下一个字符,以决定如何处理当前字符。然后就需要方便地将刚查看的字符
送回,以便下一次调用g e t c时返回该字符。
对应的输出:
putc, fputc, putchar
 
每次一行的I/O
gets, fgets
char * fgets(char * s,int size,FILE * stream);
char *gets(char *s);
fgets从文件读入, gets从表中输入读入。
fgets()用来从参数 stream所指的文件内读入字符并存到参s所指的内存空间,直到出现换行字符、读到文件尾或是已读了 size-1个字符为止,最后会加上 NULL作为字符串结束。换行符也包含在字符串中。去除换行符:s[strlen(s)-1]=0;
gets会自动删除新行符。建议不要使用gets,因为容易造成缓冲区溢出。
对应的输出:
fputs 之后不会输出一个新行符
puts 之后会输出一个新行符
 
指定长度的I/O
上面的API主要用来处理文本文件,对于结构类型的内容或结构中含有null字符或换行符的情况,不好处理。这就用到下面两个API。
size_t fread(void * ptr,size_t size,size_t nmemb,FILE * stream);
size_t fwrite(const void * ptr,size_t size,size_t nmemb,FILE * stream);
两个函数都返回实际读写的对象数。
 
定位流
long ftell(FILE * stream);
int fseek(FILE * stream,long offset,int whence);
void rewind(FILE * stream);  相当于fseek(stream, 0, SEEK_SET);
下面两个的可移植性会好些
int  fgetpos(FILE *, fpos_t *);
int fsetpos(FILE *, const fpos_t *);
 
格式化I/O
printf将格式化数据写到标准输出,fprintf写至指定的流,sprintf将格式化的字符送入数组buf中。
sprintf在该数组的尾端自动加一个null字节,但该字节不包括在返回值中。
int fprintf(FILE * stream, const char * format,.......);
int sprintf( char *str,const char * format,.........);
int printf(const char * format,.......);
 
对应的三个格式化输入
int fscanf(FILE * stream, const char * format,.......);
int sscanf( char *str,const char * format,.........);
int scanf(const char * format,.......);
标准I/O的实现细节
在UNIX上,标准I/O是调用UNIX的I/O来实现的,类似
fread()
{
  ……
  read()
  ……
}
每个流都有对应的文件描述符
可以调用int fileno(FILE *fp)来获得,如果要调用dup或fcntl等函数,需要用到这个转换。
 
创建临时文件
FILE *tmpfile(void) -- linux下可用
char *tmpnam(const char *)  -- linux下不可用
char *tempnam(const char *dir, const char *prefix); -- linux下不可用
 
char * mktemp(char * template); -- linux下可用
参数 template所指的文件名称字符串中最后六个字符必须是 XXXXXX。
linux下的一个例子:
char template[ ]="aaaaa-XXXXXX"; 
mktemp(template);
或 mkdtemp(template);

转载地址:http://zqppx.baihongyu.com/

你可能感兴趣的文章
微信小程序如何像webview一样加载html5网页
查看>>
apache和nginx的区别
查看>>
CentOs6.5系统下MySQL-5.7.19安装
查看>>
Raid 简单说明
查看>>
网络犯罪如何取证
查看>>
ms sql convert的使用细节
查看>>
Linux自学笔记——Centos7系统之systemd
查看>>
将博客搬至51CTO
查看>>
精通Java设计模式从初见到相爱之命令设计模式(15)
查看>>
linux sar命令详解
查看>>
使用Java8实现自己的个性化搜索引擎
查看>>
龙家贰少的MarkDown学习笔记
查看>>
vi 常用命令
查看>>
通过Gearman实现MySQL到Redis的数据复制
查看>>
Peer certificate cannot be authenticated with known CA certificates
查看>>
带border的百分比布局
查看>>
html input文本只读
查看>>
eclipse 自动为getter和setter添加注释
查看>>
oracle--数据库
查看>>
驰骋工作流引擎设计系列12 工作质量考核设计
查看>>