APUE-学习记录-20200624

30次阅读

共计 2035 个字符,预计需要花费 6 分钟才能阅读完成。

3.4 函数 Creat

#include <fcntl.h>
int creat(const char *path, mode_t mode); // 返回值:若成功,返回为只写打开额文件描述符;若出错,返回 -1

此函数等效于:

open(path, O_WRONLY | O_CREAT | O_TRUNC, mode);

creat 的一个不足之处是它以只写方式打开所创建的文件。在提供 open 的新版本之前,如果要创建一个临时文件,并要先写文件,然后又读该文件,则必须先调用 creat、close,然后再调用 open,现在则可用以下方式调用 open 实现:

open(path, O_EDWR | O_CREAT | O_TRUNC, mode);

3.5 函数 close

 调用 close 函数可以关闭一个打开文件。#include <unistd.h>
int close(int fd); // 返回值:若成功,返回 0;若出错,返回 -1

当关闭一个文件时,还会释放该进程加在该文件上的所有记录锁。
当一个进程终止时,内核自动关闭它所有的打开文件。

3.6 函数 lseek

每个打开文件都有一个与其相关联的“当前文件偏移量”。它通常是一个非负整数,用来度量从文件开始处计算的字节数。通常,读、写操作都从当前文件偏移量处开始,并使偏移量增加所读写的字节数。按系统默认的情况,当打开一个文件时,除非指定 O_APPEND 选项,否则该偏移量被设置为 0.

可以调用 lseek 显式为一个打开文件设置偏移量。

#include <unistd.h>
off_t lssek(int fd, off_t offset, int whence); // 返回值:若成功,返回新的文件偏移量;若出错,返回 -1

对参数 offset 的理解与参数 whence 的值有关。

  • 若 whence 是 SEEK_SET,则将该文件的偏移量设置为距文件开始处 offset 个字节。
  • 若 whence 是 SEEK_CUR,则将该文件的偏移量设置为其当前值加 offset,offset 可为正或负。
  • 若 whence 是 SEEK_END,则将该文件的偏移量设置为文件长度加 offset,offset 可为正或负。

若 lssek 成功执行,则返回新的文件偏移量,为此可以用下列方式确定打开文件的当前偏移量:

off_t currpos;
currpos = lseek(fd, 0, SEEK_CUR);

这种方法也可以用来确定所涉及的文件是否可以设置偏移量。如果文件描述符指向的是一个管道、FIFO、或网络套接字,则 lseek 返回 -1,并将 errno 设置为 ESPIPE。

Page 54 实例

#include "apue.h"
int main(void)
{if(lseek(STDIN_FILENO, 0, SEEK_CUR) == -1)
        printf("cannot seek\n");
    else
        printf("seek OK\n");
    exit(0);
}

使用终端调用此程序,则可得:

$ ./main < /etc/passwd
seek OK
$ cat < /etc/passwd | ./main
cannot seek
$ ./main < /var/spool/cron/FIFO
cannot seek

某些设备可能允许负的偏移量,因此在比较 lseek 返回值时,不要测试它是否小于 0,而要测试它是否等于 -1。

lseek 仅将当前的文件偏移量记录在内核中,并不引起任何 I / O 操作。然后,该偏移量用于下一个读或写操作。

文件偏移量可以大于文件的当前长度,在这种情况下,对该文件的下一次写将加长该文件,并在文件中构成一段空白,位于文件中但没有写过的字节都以 0 填充。

Page 55 实例

#include "apue.h"
#include "fcntl.h"

char buf1[] = "abcdefghij";
char buf2[] = "ABCDEFGHIJ";

int main(void)
{
    int fd;
    if((fd = creat("file.hole", FILE_MODE)) < 0)
        err_sys("creat error");

    if(write(fd, buf1, 10) != 10)
        err_sys("buf1 write error");
    /* offset now 10 */

    if(lseek(fd, 16384, SEEK_SET) == -1)
        err_sys("lseek error");
    /* offset noe 16384 */
    
    if(write(fd, buf2, 10) != 10)
        err_sys("buf2 write error");
    /* offset now 16394 */

    exit(0);
}

运行该程序得到:
$ ./main
$ ls -l file.hole
-rw-r–r– 1 user user 16394 Jun 24 17:29 file.hole
$ od -c file.hole
0000000 a b c d e f g h i j 0 0 0 0 0 0
0000020 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
*
0040000 A B C D E F G H I J
0040012

使用 od 命令观察该文件的实际内容。- c 标志表示以字符方式打印文件内容。~~~~

正文完
 0