【UNIX/Liux】标准I/O库【Part 3】

本文是笔者拜读《UNIX环境高级编程》第5章(标准I/O库)的学习笔记。本文的主要内容包括内存流、第5章小结和习题。文中不仅包含书中的知识点,也包括笔者的理解。

内存流

内存流相当于内存缓冲区,即用FILE指针访问内存流时,访问的不是磁盘文件,而是内存。所有的I/O都是通过在缓冲区主存之间传送字节来完成的。
创建内存流的函数有:
在这里插入图片描述
fmemopen函数允许调用者提供缓冲区用于内存流:buf参数指向缓冲区的开始位置,size参数指定了缓冲区的大小(单位是字节)。如果buf为空,fmemopen函数会分配size字节数的缓冲区,当关闭流时,缓冲区被自动释放。

fmemopen函数的mode参数和fopen函数的类似,但存在一些差别:

(1)以追加写方式打开内存流时,文件当前位置设置为缓冲区中的第一个NULL字节,如果不存在NULL字节,则文件当前位置设置为缓冲区结尾的后一个字节。如果不是以追加写方式打开内存流时,当前位置设置为缓冲区的开始位置。内存流不适合存储二进制数据

(2)如果buf参数是一个NULL指针,打开流进行只读或只写都没有任何意义,因为没法找到缓冲区的地址。

(3)任何时候需要增加流缓冲区中数据量以及调用fclosefflushfseekfseeko以及fsetpos时都会在当前位置写入一个NULL字节。
例:

#include <stdio.h>
#include <string.h>

#define N 36

int main() {
	char buf[N];
	memset(buf, 'a', N - 1);
	buf[N - 1] = '\0';
	fprintf(stdout, "%s, %ld\n", buf, (long)strlen(buf));
	
	FILE *fp = fmemopen(buf, N, "a");
	fprintf(stdout, "%s, %ld\n", buf, (long)strlen(buf));
	fclose(fp);
	
	fp = fmemopen(buf, N, "w+");
	fprintf(stdout, "%s, %ld\n", buf, (long)strlen(buf));
	fprintf(fp, "hello world");
	fprintf(stdout, "%s, %ld\n", buf, (long)strlen(buf));
	fflush(fp);	
	fprintf(stdout, "%s, %ld\n", buf, (long)strlen(buf));

	memset(buf, 'b', N - 1);	
	fprintf(stdout, "%s, %ld\n", buf, (long)strlen(buf));
	fprintf(fp, "hello world");
	fseek(fp, 0, SEEK_SET);	
	fprintf(stdout, "%s, %ld\n", buf, (long)strlen(buf));
	
	memset(buf, 'c', N - 1);	
	fprintf(stdout, "%s, %ld\n", buf, (long)strlen(buf));
	fprintf(fp, "hello world");
	fclose(fp);	
	fprintf(stdout, "%s, %ld\n", buf, (long)strlen(buf));

	return 0;
}

运行结果:
在这里插入图片描述

内存流有缓冲区
最后一个fclose(fp);为什么没有自动添加NULL字符,fprintf(fp, "hello world");难道没有增加流缓冲区中数据量

用于创建内存流的其它两个函数分别是open_memstreamopen_wmemstream
open_memstream函数创建的流是面向字节的,open_wmemstream创建的流是面向宽字节的,它们与fmemopen函数的不同在于:
(1)创建的流只能写打开。
(2)不能指定自己的缓冲区,但可以通过参数获得缓冲区地址和大小。
(3)关闭流后需要自行释放缓冲区。
(4)对流添加字节会增加缓冲区大小。

小结

大多数UNIX应用程序都使用标准I/O库。本章说明了该库提供的很多函数以及某些实现细节和效率方面的考虑。标准I/O库使用了缓冲技术,而它正是产生很多问题、引起许多混淆的部分。

习题

5.1

题目:setvbuf实现setbuf
答:

#include <stdio.h>

void mySetBuf(FILE *fp, char *buf) {
	if (!buf || fp == stderr) {
		if (setvbuf(fp, buf, _IONBF, BUFSIZ) != 0) {
			perror("set unbuffer error");
		}
	}
	else if (fp == stdin || fp == stdout) {
		if (setvbuf(fp, buf, _IOLBF, BUFSIZ) != 0) {
			perror("set linebuffer error");
		}
	}
	else {
		if (setvbuf(fp, buf, _IOFBF, BUFSIZ) != 0) {
			perror("set fullbuffer error");
		}
	}
}

5.2

题目: 使用每次一行I/Ofgetsfputs)复制文件时,若MAXLINE很小,当复制的行超过该值时会出现什么情况?对此进行解释。
答: 使用fgets一次性最多只能读MAXLINE - 1个字符(如果能读到的话,包含'\n')。如果输入文件中某行的数据过多,则需要调用多次fgets才能读完。下次调用会从当前调用中读到的最后一个字符的下一个位置开始读。

5.3

题目:printf返回0值表示什么?
答:printf的返回值是打印的字符数,如果什么也没打印出来,则返回值为0。如:

printf("");

5.4

题目: 下面的代码在一些机器上运行正确,在另外一些机器上运行出错,解释问题所在。

#include <stdio.h>

int main(void) {
	char c;
	while ((c = getchar()) != EOF) {
		putchar(c);
	}
	return 0;
}

答: EOF表示-1,当读到文件末尾时getchar会返回-1getchar的返回值类型是int,程序中被转换成了char。对有的编译器来说char相当于signed char,此时-1就是-1,程序运行正确;而对有的编译器来说char相当于unsigned char,此时-1被解释为一个很大的正数,程序运行错误。

intunsigned int类型的数据比较时,会统一转换为unsigned int,因此-1UINT_MAX“相等”,但对于char类型而言,情况似乎不同。

signed char a = -1;
unsigned char b = UCHAR_MAX;
printf("%d\n", a == b);	

signed int ai = -1;
unsigned int bi = UINT_MAX;
printf("%d\n", ai == bi);	

上面的程序会分别打印01

5.5

题目: 对标准I/O流如何使用fsync函数?
答:
在这里插入图片描述
fsync作用于fd指定的文件,等待往该文件里写数据的实际写磁盘操作结束。作用于标准输出流是:

fsync(STDOUT_FILENO);

5.6

**题目:**下列程序中,打印的提示信息没有包含换行符,程序也没有调用fflush函数,请解释输出提示信息的原因是什么?
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
**答:**在重新打印%符号前,父进程要等待子进程执行完毕,子进程执行完命令后会自带一个'\n'符。

5.7

**题目:**基于BSD系统提供了funopen的函数调用使我们可以拦截读、写、定位以及关闭一个流的调用。使用这个函数为FreeBSDMac OS X实现fmemopen

不会。。。。笔者用的是Linux系统。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值