《UNIXLinux程序设计教程》一3.2 read()和write()函数

3.2 read()和write()函数

对文件描述字进行基本输入输出操作的函数是read()和write()。

#include <unistd.h>

ssize_t read (int filedes, void * buffer, size_t nbytes);
ssize_t write (int filedes, const void * buffer, size_t nbytes);

read()从已打开的、与文件描述字filedes相连的文件中读至多nbytes个字节的数据放到buffer所指缓冲区中。它的正常返回值是实际读入的字节数,若在实际读入字节之前便遇到文件尾,其返回值为0。read()是所有读流的函数(如fgets()等)的低级原语。
数据类型ssize_t与size_t类似,UNIX系统也用它表示在一次操作中可以读写的块大小。不过与size_t不同的是,它是一个有符号整型以便能够表示错误情形的返回值。read()返回的实际读入字节数在以下几种情况下可能小于nbytes:
当读普通文件且在还未读够所请求的字节数便遇到了文件尾时。例如,当文件中只剩下30个字节便到达文件尾,而我们却企图读100个字节时,read()将返回30。下一次再调用read(),它将返回0指出文件结束。
当所读的文件对应于终端设备时,通常每次至多只读一行(在第9章将看到如何改变这种约定)。
当从网络读时,网络的内部缓冲可能导致读入的字节数少于所请求的字节数。
当被信号中断时(7.9.3节),若已读入若干字节,则返回已读的字节数;否则返回–1,并置errno为EINTR。
对于普通文件或其他可以定位的文件,read()从文件的当前位置开始读数据;在成功返回之前,文件位置增加实际已读的字节数。
read()调用成功的返回值大于等于0,遇到文件尾返回0(除非nbytes之值为0)。若在文件结束时继续调用read(),它将仍然返回0并且无其他动作。
write()将buffer的前nbytes个字节写到与描述字filedes相连的文件。buffer中的数据不必是字符串,空字符同其他字符一样写出。
write()的返回值是实际写出的字节数,正常情况下它等于nbytes,但也可能小于nbytes(例如,被写的物理介质满时)。
例3-1 程序3-1是open()、close()、read()和write()的示例程序,它是用这些低级I/O函数实现连接两个文件的另一个版本。

由于每次调用read()和write()都会使得系统访问磁盘,因此在读写时指定较大的缓冲很重要,否则程序的运行效率将会很低。这个例子中指定缓冲大小为1024字节,建议你指定不同的缓冲大小来运行这个程序,从中仔细体会缓冲大小对效率的影响。

时间: 2024-11-01 07:33:35

《UNIXLinux程序设计教程》一3.2 read()和write()函数的相关文章

《UNIXLinux程序设计教程》一导读

前言 十年前,我们出版了<UNIX程序设计教程>(清华大学出版社).十年来,影响UNIX编程接口的规范和标准发生了较大变化,当时写书参照的"Single UNIX Specification 2"现在已发展到了"Single UNIX Specification 4",而若干分离独立的规范和标准,包括Single UNIX Specification,现在都已经统一在POSIX.1-2008标准之下.同时,随着Linux系统的成熟和发展,UNIX系统已不

《UNIXLinux程序设计教程》一1.2 标准

1.2 标准 UNIX变体的激增产生了许多兼容性问题,特别是各种商业UNIX变体的出现使情况变得更加复杂.系统V和BSD在许多方面不同-它们有不同且互不兼容的物理文件系统.网络机制和虚拟内存结构.这些不同中有一些限制在内核设计和实现上,但另一些出现在程序设计接口层,这导致没有一个复杂的应用程序能够不加修改地同时运行于系统V和BSD系统.另一方面,商业变体常常带有各自的增值特征,应用程序员常常搞不清它们.结果,为了保证程序在各种不同的UNIX上都能工作,程序员不得不付出极大的努力.这种情形导致了对

《UNIXLinux程序设计教程》一第1章 UNIX导论

第1章 UNIX导论UNIX是一个"历史悠久"的操作系统.在开始讲述UNIX环境程序设计方法之前,我们先回顾UNIX的诞生.成长和发展历程,介绍UNIX发展过程中出现的若干标准.回顾UNIX的发展历史,有助于我们了解它具有如此强大生命力的原因,并把握它未来的发展方向:了解UNIX的标准,可以使我们理解和区分UNIX的不同实现与版本之间的区别,并编写出可移植性更好的程序.随后,作为后继章节的基础,本章将讲述UNIX的一些基本概念,并介绍与UNIX全系统有关的一些内容,例如系统信息.系统能

《UNIXLinux程序设计教程》一1.1 UNIX简史

1.1 UNIX简史 UNIX早在MS DOS.Windows出现之前就已经诞生了,到现在已有四十多年的历史. 1.1.1 UNIX的诞生 1965年麻省理工学院的MAC课题组和通用电气公司一起启动了一个项目-开发一个新的称为Multics的多用户.交互式操作系统.Multics的目的是向大用户团体提供同时计算和存储的能力.在当时批处理系统为主流的情况下,这是一个创新的概念.此后不久,贝尔实验室的计算科学研究中心也加入了这一计划.但在1969年,这个研究组认为开发Multics需要更长的时间,于

《UNIXLinux程序设计教程》一2.8 格式I/O

2.8 格式I/O 前几节介绍的流I/O函数除了以字符或行方式进行读写外,并不对数据进行解释,但在很多时候应用都会需要对输入输出数据进行解释,因为数据在计算机内的表示和人们可读的形式是不同的.数据在计算机内是二进制形式,在计算机外部常常为正文形式.例如,十进制数12在计算机内部的32位二进制表示是:00000000000000000000000000001100.当这个数在打印机上输出或者在终端屏幕上显示时,必须转换为字符'1'和'2',它们的ASCII编码分别为00110001和0011001

《UNIXLinux程序设计教程》一3.10 思考与练习

3.10 思考与练习 打开文件时,如果希望总是创建一个新文件,应当使用什么标志?如果希望每次写出的数据都实际写到物理存储设备,应当使用什么标志? 程序3-1中,存放读写数据的缓冲区大小为1024字节.请在你的机器上指定不同大小的缓冲区来运行这个程序,仔细体会缓冲区大小对程序效率的影响. open()调用成功总是返回当前可用的编号 的描述字.对同一个文件用不同open()打开的文件描述字具有 的文件位置,由dup()重复的文件描述字具有 的文件位置. 编写一个程序打印出指定文件的文件状态标签. 用

《UNIXLinux程序设计教程》一1.4 系统库

1.4 系统库 系统库给应用程序提供编译好的标准函数和系统调用函数的目标代码,这些代码在连接时与应用程序的目标代码装配在一起形成一个完整的可执行程序.UNIX系统库由许多专门的库组成,如C标准库.数学库.线程库.实时库等.本书将要介绍的函数和系统调用基本上都包含在系统库的C标准库中,也有部分包含在线程库和实时库中. C标准库不仅包含了C标准规定的函数(不包括科学计算函数以及国际化和宽字符函数),而且包含了POSIX标准定义的大部分编程接口函数.当我们编译C源程序时,编译器会自动地连接C标准库.但

《UNIXLinux程序设计教程》一2.5 文件定位

2.5 文件定位 读写文件过程中,有时会需要读某个特定位置的内容.例如,对于那种由固定大小的记录组成并能用整数索引来引用这些记录的文件,为访问其中某个特定的记录,最快捷的方法是直接定位至该记录位置进行读写,而不必一个一个地顺序跳过之前不需要的记录.为此,我们需要能够随意定位文件的位置,即随机地读写文件的任何部分. 标准I/O库提供了如下两组对随机文件进行定位的函数,用它们可以随机地读写文件的任何部分. #include <stdio.h> long int ftell(FILE *stream

《UNIXLinux程序设计教程》一2.7 流缓冲

2.7 流缓冲 每一个流都有一个输入输出缓冲区.写入流的字符并不立即写到文件中,而是先在缓冲区中聚集为一块,然后异步地以块为单位传送到文件.类似地,从流读出的字符也不是逐个地从文件中读出,而是以块为单位从文件读到缓冲区,然后从缓冲区传送给进程.这种处理方式称为缓冲. 采用缓冲的目的是为了减少调用低级I/O函数(如read()和write())的次数,因为这些真正读写文件的函数是系统调用,它们是较费时间的操作.例如,对于存储在硬盘上的文件,当进程用read()或write()读写数据时,设备驱动程

《UNIXLinux程序设计教程》一2.10 思考与练习

2.10 思考与练习 打开文件的实质是什么? 从应用的角度看,UNIX系统中程序与文件建立连接有几种机制?流与文件描述字有什么区别? 什么是文件位置?它起什么作用? 系统为每一个进程自动打开的输入输出流有哪些?它们对应的名字是什么? 写"r"方式打开的文件会发生什么情况?读"w"方式打开的文件呢?建议你编写一个这样的程序试试. 按读写数据的粒度分,有几类流输入输出函数? 为什么说gets()是危险的函数? 程序2-3说明了fgets()和gets()的不同.运行该程