一些 C C++ POSIX 的 IO 操作总结

  • 文件-内存之间
  • 内存-内存之间
  • POSIX 有无缓冲的 IO 操作

对文件的操作,读文件至内存,从内存写至文件

// 读文件至内存buf中
void Fread()
{
char buf[BUFSIZE];
size_t flen = 0;
FILE *fp = fopen("./foo", "r+"); flen = fread(buf, sizeof(char), BUFSIZE, fp); // fp -> buf
printf("read len: %lu\n", flen);
printf("Fread text:\n%s\n\n", buf);
fclose(fp);
} // 将内存 buf 中的内容写入打开的文件 fp 中
void Fwrite()
{
char buf[BUFSIZE] = BUFTEXT;
size_t flen = 0;
FILE *fp = fopen("./foo", "w+");
flen = fwrite(buf, sizeof(char), strlen(buf), fp); // buf -> fp
printf("Fwrite write len: %lu\n\n", flen);
fclose(fp);
} // 将fp 换为 stdin 输入至 buf 中
void Fscanf()
{
char buf[BUFSIZE];
size_t flen = 0;
FILE *fp = fopen("./foo", "r+");
while(fscanf(fp, "%s", buf) != EOF) // fp -> buf, terminated by blank char
printf("Fscanf text:\n%s\n", buf);
printf("\n");
fclose(fp);
} // C++ 文件流,输入至内存中,由于流遇到空格等空白符停止所以需循环将流中的内容全部输入至
// 内存(string)中,直至返回 EOF
void Ifstream()
{
ifstream ifs("./foo");
string buf;
while (ifs >> buf) // ifs -> buf
cout <<"Ifstream: " << buf << endl; // char *str = new char[100];
// ifs.read(str, 10);
// whlile (getline(ifs, buf))
// cout << "Ifstream: " << buf << endl;
ifs.close();
cout << endl;
} // fp 换为 stdout, buf 中的内容写入至 fp 中
void Fprintf()
{
char buf[BUFSIZE] = "ioi ioi";
int flen = 0;
FILE *fp = fopen("./foo", "w+");
flen = fprintf(fp, "%s", buf); // buf -> fp
printf("Fprintf len: %d\n", flen);
fclose(fp);
} void Ofstream()
{
ofstream ofs("./foo", ios::app | ios::out);
string buf = "quick,,time go on";
// ofs << buf;
ofs.write(buf.c_str(), buf.size());
ofs.close();
}

字符流的操作,内存至内存

// 文件流 stdio 换为 str, 格式化输入
void Sscanf()
{
char str[BUFSIZE] = BUFTEXT;
char buf[BUFSIZE];
sscanf(str, "%s", buf); // str -> buf
printf("Sscanf text:\n%s\n\n", buf);
} // istringstream 对象 iss 绑定 buf,作为输入流输入至 word(string)
void Istringstream()
{
string buf = "hello world";
string word; istringstream iss(buf); // buf -> iss
while (iss >> word) // iss -> word, repeatedly
cout << "Istringstream: " << word << endl;
} // 与 sscanf() 类似, 格式化输出
void Sprintf()
{
char str[BUFSIZE] = BUFTEXT;
char buf[BUFSIZE];
sprintf(buf, "%s", str); // str -> buf
printf("Sprintf text:\n%s\n\n", buf);
} // 与 istringstream 类似
void Ostringstream()
{
string buf = BUFTEXT;
string word; ostringstream oss;
oss << buf; // buf -> oss
word = oss.str(); // oss -> word
cout << "Ostringstream: " << word << endl;
}

Linux 有无缓冲的 IO操作 (参考 UNP)

// 无缓冲, 普通的 read() write() 并不难保证完成所需要传送的字符数
ssize_t rio_readn(int fd, void *usrbuf, size_t n)
{
size_t nleft = n;
ssize_t nread;
char *bufp = usrbuf; while (nleft > 0) {
if ((nread = read(fd, bufp, nleft)) < 0) {
if (errno == EINTR) // interrupted by sig handler return
nread = 0; // call read() again
else
return -1; // errno set by read()
} else if (nread == 0)
break; // EOF
nleft -= nread;
bufp += nread;
} return (n - nleft);
} // not buff read
ssize_t rio_writen(int fd, void *usrbuf, size_t n)
{
size_t nleft = n;
ssize_t nwritten;
char *bufp = usrbuf; while(nleft > 0) {
if ((nwritten = write(fd, bufp, nleft)) <= 0) {
if (errno == EINTR)
nwritten = n;
else
return -1;
}
nleft -= nwritten;
bufp += nwritten;
} return n;
} // IO 缓冲区结构体
typedef struct {
int rio_fd;
int rio_cnt;
char *rio_bufptr;
char rio_buf[RIO_BUFSIZE];
} rio_t;
// 与 read() 语义相同
// 如果缓冲区为空会通过调用 read() 填满它
static ssize_t rio_read(rio_t *rp, char *usrbuf, size_t n)
{
int cnt; while (rp->rio_cnt <= 0) { // refill if buf is empty
rp->rio_cnt = read(rp->rio_fd, rp->rio_buf, sizeof(rp->rio_buf));
if (rp->rio_cnt < 0) {
if (errno != EINTR)
return -1;
} else if (rp->rio_cnt == 0) { // EOF
return 0;
} else {
rp->rio_bufptr = rp->rio_buf; // reset buffer ptr
}
} // copy min(n, rp->cnt) bytes from internal buf to user buf
cnt = n;
if (rp->rio_cnt < n)
cnt = rp->rio_cnt;
memmove(usrbuf, rp->rio_bufptr, cnt);
rp->rio_bufptr += cnt;
rp->rio_cnt -= cnt; return cnt;
}

一点小结

  • 本质是操作对象的变化
// C
printf() -- stdout scanf() -- stdin
fprintf() -- FILE* fscanf() -- FILE*
sprintf() -- char* sscanf() -- str
// C++ 与上类似, 都具有缓冲区
ostream istream
ofstream ifstream
ostringstream istringstream
  • 带有缓冲区的 rio_read() 有时效率很高,因为不必一直在内核-用户态之间切换

C C++ POSIX 的一些 IO 操作的更多相关文章

  1. python之协程与IO操作

    协程 协程,又称微线程,纤程.英文名Coroutine. 协程的概念很早就提出来了,但直到最近几年才在某些语言(如Lua)中得到广泛应用. 子程序,或者称为函数,在所有语言中都是层级调用,比如A调用B ...

  2. Linux学习记录--文件IO操作相关系统编程

    文件IO操作相关系统编程 这里主要说两套IO操作接口,各自是: POSIX标准 read|write接口.函数定义在#include<unistd.h> ISO C标准 fread|fwr ...

  3. Linux文件IO操作

    来源:微信公众号「编程学习基地」 目录 文件操作 Linux文件类型 Linux文件权限 修改文件权限 Linux error 获取系统调用时的错误描述 打印错误信息 系统IO函数 open/clos ...

  4. [.NET] 利用 async & await 进行异步 IO 操作

    利用 async & await 进行异步 IO 操作 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/6082673.html  序 上次,博主 ...

  5. 文件IO操作..修改文件的只读属性

    文件的IO操作..很多同行的IO工具类都是直接写..但是如果文件有只读属性的话..则会写入失败..所以附加了一个只读的判断和修改.. 代码如下: /// <summary> /// 创建文 ...

  6. JAVASE02-Unit08: 文本数据IO操作 、 异常处理

    Unit08: 文本数据IO操作 . 异常处理 * java.io.ObjectOutputStream * 对象输出流,作用是进行对象序列化 package day08; import java.i ...

  7. JAVASE02-Unit07: 基本IO操作 、 文本数据IO操作

    基本IO操作 . 文本数据IO操作 java标准IO(input/output)操作 package day07; import java.io.FileOutputStream; import ja ...

  8. IO操作概念。同步、异步、阻塞、非阻塞

    “一个IO操作其实分成了两个步骤:发起IO请求和实际的IO操作. 同步IO和异步IO的区别就在于第二个步骤是否阻塞,如果实际的IO读写阻塞请求进程,那么就是同步IO. 阻塞IO和非阻塞IO的区别在于第 ...

  9. Java基础复习笔记系列 七 IO操作

    Java基础复习笔记系列之 IO操作 我们说的出入,都是站在程序的角度来说的.FileInputStream是读入数据.?????? 1.流是什么东西? 这章的理解的关键是:形象思维.一个管道插入了一 ...

随机推荐

  1. 解决myeclipse中struts2 bug问题包的替换问题

    由于struts2的bug问题,手工替换还是比較麻烦.但即便是最新的myeclipse2014也没有替换最新的struts2包,研究了一天,最终找到了解决的方法.下面就解决方法与大家分享一下. 1.在 ...

  2. tiny4412 裸机程序 五、控制icache【转】

    本文转载自:http://blog.csdn.net/eshing/article/details/37115411 版权声明:本文为博主原创文章,未经博主允许不得转载.   目录(?)[+]   一 ...

  3. 枚举类enum的values()方法

    value()方法可以将枚举类转变为一个枚举类型的数组,因为枚举中没有下标,我们没有办法通过下标来快速找到需要的枚举类,这时候,转变为数组之后,我们就可以通过数组的下标,来找到我们需要的枚举类.接下来 ...

  4. E20170916-hm

    sassy   adj. 无礼的; 漂亮的; <非正,美> <贬>粗鲁的; <褒>时髦的; digest vt. 消化; 整理; compressor  n. 压气 ...

  5. P3398仓鼠(LCA)

    题目描述 小仓鼠的和他的基(mei)友(zi)sugar住在地下洞穴中,每个节点的编号为1~n.地下洞穴是一个树形结构.这一天小仓鼠打算从从他的卧室(a)到餐厅(b),而他的基友同时要从他的卧室(c) ...

  6. Django day32 跨域问题,创建vue项目,axios的使用

    一:跨域问题 1.同源策略(浏览器的安全策略) 只允许当前页面朝当前域下发请求,如果向其他域发请求,请求可以正常发送,数据也可以拿回,但是被浏览器拦截了 2.cors:只要服务器实现了CORS,就可以 ...

  7. 【转】深入理解Java多态原理

    之前一直知道多态是什么东西,平时敲代码也经常用到多态,但一直没有真正了解多态底层的运行机制到底是怎么样的,这两天才研究明白点,特地写下来,跟各位同学一起进步,同时也希望各位大神指导和指正. 多态的概念 ...

  8. 【POJ1845】Sumdiv(数论/约数和定理/等比数列二分求和)

    题目: POJ1845 分析: 首先用线性筛把\(A\)分解质因数,得到: \[A=p_1^{a_1}*p_2^{a_2}...*p_n^{a_n} (p_i是质数且a_i>0) \] 则显然\ ...

  9. CSS------选择器-----------选择器的分组、属性选择器

    /*!--选择器的分组--*/ .groupDiv h1,h2,h3,h4{ color: #000000; } /*------------------------属性选择器--*/ [title] ...

  10. 每天学点Linux命令之 vi 命令

    来学一个vi的命令.要完成的是在一个只读文件中,删掉一行,然后插入两行. 那只读文件你要修改,用sudo vi 总可以了吧.首先 vi命令进入编辑模式. 在非插入模式中: h 光标左移  l 光标右移 ...