C语言KR圣经笔记 8.3 open,creat,close,unlink 8.4随机访问-lseek

2024-02-23 14:28

本文主要是介绍C语言KR圣经笔记 8.3 open,creat,close,unlink 8.4随机访问-lseek,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

8.3 open, creat, close, unlink

除了标准输入、标准输出和标准错误之外,如果你要读写文件,就必须显式地打开它们。有两个系统调用做这件事,open 和 creat(末尾就是没有 e 的)。

open 非常像第七章讨论的 fopen,区别在于 open 不返回文件指针,而是返回文件描述符,后者仅仅是个 int 。如果发生任何错误,则 open 返回 -1。

#include <fcntl.h>int fd;
int open(char *name, int flags, int perms);fd = open(name, flags, perms);

与 fopen 一样,name 参数是包含文件名的字符串。第二个 int 参数 flags 是用来指定如何打开文件的;主要的值有

O_RDONLY     只读
O_WRONLY    只写
O_RDWR        读写

在 System V UNIX 系统上,这些常量定义在 <fcntl.h> 中,而在伯克利(BSD)版本的系统上,它们定义在 <sys/file.h> 中。

若为了读取而打开已存在的文件,要写成

fd = open(name, O_RDONLY, 0);

接下来在介绍 open 的使用时,我们总是把参数 perms 的值设为 0。

试图 open 一个不存在的文件是错误的。creat 系统调用被用于创建新文件,或者重写旧文件。

int creat(char *name, int perms);fd = creat(name, perms);

如果能够创建文件,则 creat 返回一个文件描述符,否则返回 -1。如果文件已存在,则 creat 将其大小截断成 0,这样就丢弃了文件之前的内容;creat 已存在的文件不是错误。

如果文件不存在,则 creat 使用 perms 参数所指定的权限来创建文件。在 UNIX 系统中,有 9 个比特位的权限信息与文件关联,分别控制文件属主,属主所在的用户组,以及其他用户的读,写和执行。因此,可以很容易用一个三位的八进制数来指定权限。例如,0755指定属主有读、写和执行权限,用户组和其他用户有读和执行权限。

这里以 UNIX 系统中用于拷贝文件的 cp 程序的一个简单版本为例。我们这个版本的 cp 只能拷贝一个文件,也不允许第二个参数为目录,还写死了文件权限,没有从原文件拷贝过来。

#include <stdio.h>
#include <fcntl.h>
#include "syscalls.h"
#define PERMS 0666    /* 属主,组和其他人都允许读和写 */void error(char *, ...);/* cp: 把 f1 拷贝到 f2 */
main(int argc, char *argv[])
{int f1, f2, n;char buf[BUFSIZE];if (argc != 3)error("Usage: cp from to");if ((f1 = open(argv[1], O_RDONLY,0))  == -1)error("cp: can't open %s", argv[1]);if ((f2 = creat(argv[2], PERMS)) == -1)error("cp: can't create %s, mode %03o", argv[2], PERMS);while ((n = read(f1, buf, BUFSIZ)) > 0)if (write(f2, buf, n) != n)error("cp: write error on file %s", argv[2]);return 0;
}

这个程序使用固定权限 0666 来创建输出的文件。使用后面 8.6 节所述的 stat 系统调用,我们可以确定已存在文件的权限,并将同样的权限赋给要拷贝的文件。

注意 error 函数调用方式很像 printf,也有一个可变参数列表。下面 error 的实现代码说明了如何使用 printf 家族的另一个成员。标准库函数 vprintf 与 printf 类似,区别在于可变参数列表被换成了单个参数,且这个参数已经使用 va_start 宏初始化过了。类似的,vfprintf 和 vsprintf 对应 fprintf 和 sprintf。

#include <stdio.h>
#include <stdarg.h>/* error:打印错误信息并结束程序 */
void error(char *fmt, ...)
{va_list args;va_start(args, fmt);fprintf(stderr, "error: ");vfprintf(stderr, fmt, args);fprintf(stderr, "\n");va_end(args);exit(1);
}

一个程序能够同时打开的文件数量会有限制(通常大概是20)。因此,任何想要处理很多文件的程序必须准备好重用文件描述符。函数 close(int fd) 将文件描述符与一个打开文件之间的关联断开,并释放文件描述符以供其他文件使用;它对应标准库中的 fclose,区别在于不进行缓冲刷新。通过 exit 退出程序,或者从 main 程序退出,会关闭所有已打开的文件。

函数 unlink(char *name) 从文件系统中删除文件 name。它对应标准库函数 remove。

练习8-1、重写第七章的 cat 程序,使用 read、write、open 和 close,而不是它们对应的标准库函数。做试验来确定这两个版本的相对执行速度。

8.4 随机访问-lseek


输入和输出通常是按顺序的:每次 read 和 write 都正好发生在上一次读写的文件位置之后。不过若有需要,文件可以按任意顺序来读写。系统调用 lseek 提供了一种在文件中移动,但不读取或写入任何数据的方式:

long lseek(int fd, long offset, int origin);

lseek 把文件描述符为 fd 的文件的当前位置设为由 origin 所指定的 offset 偏移位置。后续的读写会从该位置开始。origin 可以是 0、1、2,分别表示 offset 偏移量从文件开头,或从当前位置,或从文件末尾计算。例如,要从文件末尾添加(UNIX shell 中的 >> 重定向,或者 fopen 的 "a"),则在写文件之前要移到文件末尾:

lseek(fd, 0L, 2);

要回到文件开头(“倒回”):

lseek(fd, 0L, 0);

注意参数 0L;如果 lseek 正确声明的话,也能写成 (long) 0 或 仅一个 0。

有了 lseek,就可能把文件当作一个大数组来对待,代价是访问较慢。例如,下面的函数从文件
的任意位置读取任意数量的字节。它返回读到的数量,若出错则返回 -1。

#include "syscalls.h"/* get:从pos位置读取n个字节 */
int get(int fd, long pos, char *buf, int n)
{if (lseek(fd, pos, 0) >= 0)    /* 移动到pos位置 */return read(fd, buf, n);elsereturn -1;
}

lseek 的返回值是一个 long,给出了文件的新位置,若出错则返回 -1。标准库函数 fseek 与 lseek 类似 ,区别在于它的第一个参数是 FILE *,且在出错时返回非0。


 

这篇关于C语言KR圣经笔记 8.3 open,creat,close,unlink 8.4随机访问-lseek的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/738939

相关文章

Python中随机休眠技术原理与应用详解

《Python中随机休眠技术原理与应用详解》在编程中,让程序暂停执行特定时间是常见需求,当需要引入不确定性时,随机休眠就成为关键技巧,下面我们就来看看Python中随机休眠技术的具体实现与应用吧... 目录引言一、实现原理与基础方法1.1 核心函数解析1.2 基础实现模板1.3 整数版实现二、典型应用场景2

C语言中的数据类型强制转换

《C语言中的数据类型强制转换》:本文主要介绍C语言中的数据类型强制转换方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C语言数据类型强制转换自动转换强制转换类型总结C语言数据类型强制转换强制类型转换:是通过类型转换运算来实现的,主要的数据类型转换分为自动转换

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件

C语言实现两个变量值交换的三种方式

《C语言实现两个变量值交换的三种方式》两个变量值的交换是编程中最常见的问题之一,以下将介绍三种变量的交换方式,其中第一种方式是最常用也是最实用的,后两种方式一般只在特殊限制下使用,需要的朋友可以参考下... 目录1.使用临时变量(推荐)2.相加和相减的方式(值较大时可能丢失数据)3.按位异或运算1.使用临时

使用C语言实现交换整数的奇数位和偶数位

《使用C语言实现交换整数的奇数位和偶数位》在C语言中,要交换一个整数的二进制位中的奇数位和偶数位,重点需要理解位操作,当我们谈论二进制位的奇数位和偶数位时,我们是指从右到左数的位置,本文给大家介绍了使... 目录一、问题描述二、解决思路三、函数实现四、宏实现五、总结一、问题描述使用C语言代码实现:将一个整

C语言字符函数和字符串函数示例详解

《C语言字符函数和字符串函数示例详解》本文详细介绍了C语言中字符分类函数、字符转换函数及字符串操作函数的使用方法,并通过示例代码展示了如何实现这些功能,通过这些内容,读者可以深入理解并掌握C语言中的字... 目录一、字符分类函数二、字符转换函数三、strlen的使用和模拟实现3.1strlen函数3.2st

Go语言中最便捷的http请求包resty的使用详解

《Go语言中最便捷的http请求包resty的使用详解》go语言虽然自身就有net/http包,但是说实话用起来没那么好用,resty包是go语言中一个非常受欢迎的http请求处理包,下面我们一起来学... 目录安装一、一个简单的get二、带查询参数三、设置请求头、body四、设置表单数据五、处理响应六、超

使用Dify访问mysql数据库详细代码示例

《使用Dify访问mysql数据库详细代码示例》:本文主要介绍使用Dify访问mysql数据库的相关资料,并详细讲解了如何在本地搭建数据库访问服务,使用ngrok暴露到公网,并创建知识库、数据库访... 1、在本地搭建数据库访问的服务,并使用ngrok暴露到公网。#sql_tools.pyfrom

C语言中的浮点数存储详解

《C语言中的浮点数存储详解》:本文主要介绍C语言中的浮点数存储详解,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、首先明确一个概念2、接下来,讲解C语言中浮点型数存储的规则2.1、可以将上述公式分为两部分来看2.2、问:十进制小数0.5该如何存储?2.3 浮点

Javascript访问Promise对象返回值的操作方法

《Javascript访问Promise对象返回值的操作方法》这篇文章介绍了如何在JavaScript中使用Promise对象来处理异步操作,通过使用fetch()方法和Promise对象,我们可以从... 目录在Javascript中,什么是Promise1- then() 链式操作2- 在之后的代码中使