函数creat
除了用open创建新文件之外,还可以使用creat函数创建一个新文件。
#include <fcntl.h> int creat(const char * path, mode_t mode); |
---|
注意:此函数等效于:
open(path, O_WRONLY | O_CREAT | O_TRUNC, mode);
creat的一个不足之处是它以只写方式打开所创建的文件。在提供open的新版本之前,如果要创建一个临时文件,并要先写该文件,然后又读该文件,则必须先调用creat、close,然后再调用open。现在则可用下列方式调用open实现:
open(path, O_RDWR | O_CREAT | O_TRUNC, mode);
函数close
可调用close函数关闭一个打开文件
#include <unistd.h> int close (int fd); |
---|
关闭一个文件时还会释放该进程加在该文件上的所有记录锁。
当一个进程终止时,内核自动关闭它所有的打开文件。
函数lseek
每个打开文件都有一个与其相关联的“当前文件偏移量”(current file offset)。它通常是一个非负整数,用以度量从文件开始出计算的字节数。通常,读、写操作都从当前文件偏移量处开始,并使偏移量增加所读写的字节数。
可以调用lseek显式的为一个打开文件设置偏移量。
#include <unistd.h> off_t lseek(int fd, off_t offset, int whence) |
---|
对参数offset的解释与参数whence的值有关
whence | 解释 |
---|---|
SEEK_SET | 偏移量设置为距文件开始处offset个字节 |
SEEK_CUR | 偏移量设置为当前位置加上offset个字节,offset可正可负 |
SEEK_END | 偏移量设置为文件大小加上offset个字节,offset可正可负 |
若lseek成功执行,则返回新的文件偏移量,为此可以用下列方式确定打开文件的当前偏移量:
off_t currpos;
currops = lseek(fd, 0, SEEK_CUR);
lseek仅将当前的文件偏移量记录在内核中,它并不引起任何的I/O操作。然后,该偏移量用于写一个读或者写操作。
文件偏移量可以大于该文件的当前长度,在这种情况下,对该文件的下一次写将加长该文件,并在文件中构成一个空洞,这一点事允许的。位于文件中但没有写过的字节都被读为0.
文件的空洞并不要求在磁盘上占用储存区。具体处理方式与文件系统的实现有关,当定位到超出文件尾端之后写时,对于新写的数据需要分配磁盘块,但是对于原文件尾端和新开始写位置之间的部分则不需要分配磁盘块。
#include <stdio.h>
#include <fcntl.h>
char buf1[] = "abcdefghij";
char buf2[] = "ABCDEFGHIJ";
int main(void)
{
int fd;
if((fd = open("file.hole", O_RDWR | O_CREAT | O_TRUNC)) < 0)
{
perror("creat error");
exit(-1);
}
if(write(fd, buf1, 10) != 10)
{
perror("write error");
exit(-1);
}
if(lseek(fd, 16384, SEEK_SET) == -1)
{
perror("seek error");
exit(-1);
}
if(write(fd, buf2, 10) != 10)
{
perror("write error");
exit(-1);
}
exit(0);
}
shell演示
$ ./a.out
$ ls -l file.hole
-r-Sr-s–T 1 root root 16394 Apr 13 14:30 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
*
0002000 A B C D E F G H I J
0002012
注意:尽管可以实现64位文件偏移量,但是能否创建一个大于2GB的文件则依赖于底层文件系统的类型。