3.1 当读/写磁盘文件时,本章中描述的函数是否具有缓冲机制?请说明原因。

  1. 3.1 所有的磁盘 I/O 都要经过内核的块缓冲区(也称为内核的缓冲区高速缓存),唯一例
  2. 外的是对原始磁盘设备的 I/O,但是我们不考虑这种情况。Bach[1986]的第 3 章描述
  3. 了这种缓冲区高速缓存的操作。既然 read write 的数据都要被内核缓冲,那么术
  4. 语“不带缓冲的 I/O”指的是用户的进程中对这两个函数不会自动缓冲,每次 read
  5. write 就要进行一次系统调用。
  1. 3.2 编写一个与 3.12 节中 dup2 功能相同的函数,要求不调用 fcntl 函数,并且要有正确的出错处理。

int dup2(int filedes, int filedes2); 将filedes2对应的打开文件置为filedes对应的打开文件,之后就可以使用filedes2操作filedes打开的文件。

由于要复制一个文件描述符,而且不能使用fcntl(),那么,唯一能使用的就是dup()了:

int dup(int filedes); 复制一个当前可用的最小的文件描述符,使它对应的打开文件是filedes对应的打开文件。

这里采用的主要思想是:dup()返回的是当前可用的最小的文件描述符,那么就可以使用遍历,从dup(filedes)开始遍历,知道返回的文件描述符等于filedes2。

下面介绍主要流程:

当filedes2是无效文件描述符时,输出错误信息,那么什么样的是无效文件描述符呢?就是小于0或者大于进程可以打开的最大文件数时,文件描述符是无效的。

然后就可以分为三种情况:

filedes == filedes2: 直接返回filedes2

filedes > filedes2: 关闭filedes2,再进行dup(filedes)返回值应该就是filedes

filedes < filedes2: 从dup(filedes)开始遍历,直到返回值等于filedes2,然后关闭filedes2,再dup(filedes)返回值应该就是filedes

后面两种情况可以结合一下,看看代码,代码参考了APUE习题 3.2 浅析

  1. #include <unistd.h>
  2. #include <fcntl.h>
  3. #include <stdio.h>
  4.  
  5. #define MAXN 4096
  6. #define EXIT_SUCC 0
  7. #define EXIT_FAIL -1
  8.  
  9. int dup2_func(int filedes, int filedes2)
  10. {
  11. int i = ;
  12. int n = ;
  13. int top = ;
  14. int stack[MAXN];
  15.  
  16. if((filedes2 > OPEN_MAX) || (filedes2 < )) {
  17. printf("invalid filedes2!n");
  18. return EXIT_FAIL;
  19. }
  20.  
  21. if(filedes == filedes2) {
  22. return filedes2;
  23. }
  24.  
  25. while((n = dup(filedes)) < filedes2) {
  26. if(n == -) {
  27. printf("System can not make a filedes!n");
  28. return EXIT_FAIL;
  29. }
  30. stack[top++] = n;
  31. }
  32.  
  33. close(filedes2);
  34.  
  35. if(dup(filedes) == -) {
  36. printf("dup function error!n");
  37. return EXIT_FAIL;
  38. }
  39.  
  40. for(i = ; i < top; ++i) {
  41. close(stack[i]);
  42. }
  43.  
  44. return filedes2;
  45. }
  46.  
  47. int main(int argc, char *argv[])
  48. {
  49. int filedes, filedes2;
  50.  
  51. if(argc != ) {
  52. printf("Parameter error!n");
  53. return EXIT_FAIL;
  54. }
  55.  
  56. filedes = open(argv[], O_RDWR);
  57. if(filedes == -) {
  58. printf("Error! System cannot open %sn", argv[]);
  59. return EXIT_FAIL;
  60. }
  61.  
  62. filedes2 = atoi(argv[]);
  63. if(dup2_func(filedes, filedes2) != EXIT_FAIL) {
  64. write(filedes2, "test", sizeof("test"));
  65. }
  66.  
  67. return ;
  68. }

假设一个进程执行下面的3个函数调用:
fd1 = open(pathname, oflags);
fd2 = dup(fd1)
fd3 = open(pathname, oflags)

要画出一个表图!
我知道fd1和fd2会指向同一个文件表数据结构,但对于fd3这个就不知道怎么回事!看了一下结果,才明白,很有意思~

 
 
fd3会指向另一个文件表,但最终还是会指向与fd1同一个物理地址。

fcntl作用于fd1来说,F_SETFD命令只影响fd1的文件描述符标志;F_SETFL作用于fd1时,则影响fd1和fd2指向的文件表项。

 

3.4 在许多程序中都包含了下面一段代码:

  1. dup2(fd, 0);
  2. dup2(fd, 1);
  3. dup2(fd, 2);
  4. if(fd > 2)
  5. close(fd);

请说明if语句的必要性。

其实,首先可以看看,这几句话可以实现什么功能?

三个函数将0, 1, 2指向的文件表项赋为fd指向的文件表项,因此,这三个函数起到文件重定向的作用,使对标准输入输出出错的操作定向到fd打开的文件。

如果fd <= 2,也就是对三个标准的操作定向到其中一个的操作。

如果fd > 2,也就是对三个标准的操作定向到fd打开的文件,此时就有4个文件描述符指向fd打开的文件。

为什么要关闭fd呢?书上给的答案是:“这种情况下就需要关闭描述符3”。但没有说明为什么。

如果不关闭fd会有什么问题吗?个人认为,已经有了三个标准的操作,此时就不需要fd了,而且如果fd再操作的话容易引起混乱,纯属个人见解。

3.5 在Bourne shell、Bourne-again shell和Korn shell中,digit1 > &digit2表示要将描述符digit1重定向至描述符digit2的同一文件。请说明下面两条命令的区别。

  1. ./a.out > outfile 2>&1
  2. ./a.out 2&>1 > outfile

从左往右看

第一条命令:先将标准输出重定向到outfile,即1 > outfile,然后2>&1,将标准出错重定向到标准输出,由于标准输出已经重定向到outfile,因此,标准出错也重定向到outfile。

第二条命令:先将标准出错重定向到标准输出,然后将标准输出重定向到标准输出重定向到outfile。

结果是:

第一条命令:标准输出和标准出错都定向到outfile。

第二条命令:标准出错指向标准输出重定向之前的打开文件,标准输出定向到outfile。

文件IO 练习题的更多相关文章

  1. JAVA 基础编程练习题50 【程序 50 文件 IO】

    50 [程序 50 文件 IO] 题目:有五个学生,每个学生有 3 门课的成绩,从键盘输入以上数据(包括学生号,姓名,三门课成绩), 计算出平均成绩,将原有的数据和计算出的平均分数存放在磁盘文件&qu ...

  2. 标准io与文件io

    A: 代码重复: 语句块1: while(判断) { 语句块2: 语句块1: } 上面可以改写为: while(1) { 语句块1: if(判断) break: 语句块2: } B: 标准IO和文件I ...

  3. 文件IO函数和标准IO库的区别

    摘自 http://blog.chinaunix.net/uid-26565142-id-3051729.html 1,文件IO函数,在Unix中,有如下5个:open,read,write,lsee ...

  4. 转 漫谈linux文件IO

    在Linux 开发中,有几个关系到性能的东西,技术人员非常关注:进程,CPU,MEM,网络IO,磁盘IO.本篇文件打算详细全面,深入浅出.剖析文件IO的细节.从多个角度探索如何提高IO性能.本文尽量用 ...

  5. (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

  6. Java文件IO操作应该抛弃File拥抱Paths和Files

    Java7中文件IO发生了很大的变化,专门引入了很多新的类: import java.nio.file.DirectoryStream;import java.nio.file.FileSystem; ...

  7. Java 文件IO续

    文件IO续 File类    用来将文件和文件夹封装成对象 方便对文件和文件夹的属性信息进行操作    File对象可以作为参数传递给流的构造函数 Demo1 File的构造方法 public cla ...

  8. Java 文件IO

    文件IO Java IO    IO流用来处理设备之间的数据传输 Java对数据的操作是通过流的方式 Java用于操作流的对象都在IO包中    按操作数据分为 字节流和字符流        字符流的 ...

  9. 文件IO和标准IO

    2015.2.26 星期四,阴天 今天的内容主要是文件IO man 手册的分册: man -f open 查看那些分册中有openman 1 -- 普通的命令程序man 2 -- 系统调用man 3 ...

随机推荐

  1. 定时显示div

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" " http://www.w3.org ...

  2. python JSON处理

    概念 序列化(Serialization):将对象的状态信息转换为可以存储或可以通过网络传输的过程,传输的格式可以是JSON.XML等. 反序列化:就是从存储区域(JSON,XML)读取反序列化对象的 ...

  3. ASP.NET多用户操作相同互斥的对象

    [一篮饭特稀原创,转载请注明出自http://www.cnblogs.com/wanghafan/p/3574154.html ] 现象:公有静态变量不可用于ASP.NET多用户操作,否则该变量会被多 ...

  4. 例说C#深拷贝与浅拷贝

    一开始,先对C#深拷贝与浅拷贝知识做个简单的总结. 无论是浅拷贝与深拷贝,C#都将源对象中的所有字段复制到新的对象中.不过,对于值类型字段,引用类型字段以及字符串类型字段的处理,两种拷贝方式存在一定的 ...

  5. Oracle SQL编写注意事项

    1.SQL语句用大写的;因为Oracle总是先解析SQL语句,把小写的字母转换成大写的再执行. 2.数据表最好起别名;因为便于sql优化器快速分析. 3.尽量不要使用 insert into tabl ...

  6. Win7,8的上帝模式文件夹 GodMode.{ED7BA470-8E54-465E-825C-99712043E01C}

    新建一文件夹,命名为:“GodMode.{ED7BA470-8E54-465E-825C-99712043E01C}”, 创建后双击进入,可以看到计算机所有的配置选项信息.

  7. POJ 3666 Making the Grade (DP滚动数组)

    题意:农夫约翰想修一条尽量平缓的路,路的每一段海拔是A[i],修理后是B[i],花费|A[i] – B[i]|,求最小花费.(数据有问题,代码只是单调递增的情况) #include <stdio ...

  8. Android图形系统之Surface、SurfaceView、SurfaceHolder及SurfaceHolder.Callback开发实例

    原文:Android图形系统之Surface.SurfaceView.SurfaceHolder及SurfaceHolder.Callback之间的联系 Surface是原始图像缓冲区(raw buf ...

  9. hug and Compression Resistance

    Hugging => content does not want to grow Compression Resistance => content does not want to sh ...

  10. dd usb 启动盘制作 成功版本

    在linux系统中,使用dd命令制作启动盘成功.方法是在终端中输入命令:     dd if=/root/opensuse.iso of=/dev/sdb bs=4M     说明:      1.o ...