creat、close、lseek函数

可用creat函数创建一个新文件。
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
int creat(const char* pathname, mode_t mode) ;
返回:若成功为只写打开的文件描述符,若出错为- 1
注意,此函数等效于:
open(pathname, O_WRONLY|OCREAT|O_TRUNC, mode) ;
在早期的UNIX版本中, open的第二个参数只能是0、1或2。没有办法打开一个尚未存在的文件,因此需要另一个系统调用c reat以创建新文件。现在,open函数提供了选择项OCREAT和OTRUNC,于是也就不再需要creat函数了。
在4 . 5节中,我们将详细说明文件存取许可权,并说明如何指定mode。
creat的一个不足之处是它以只写方式打开所创建的文件。在提供open的新版本之前,如果要创建一个临时文件,并要先写该文件,然后又读该文件,则必须先调用creat,close,然后再调用open。现在则可用下列方式调用open:
open (pathname, O_RDWR|OCREAT|O_TRUNC, mode) ;

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

 

关闭一个文件时也释放该进程加在该文件上的所有记录锁。1 2 . 3节将讨论这一点。当一个进程终止时,它所有的打开文件都由内核自动关闭。很多程序都使用这一功能而不显式地用c l o s e关闭打开的文件。实例见程序1 - 2。


3.6 lseek函数
每个打开文件都有一个与其相关联的“当前文件位移量”。它是一个非负整数,用以度量从文件开始处计算的字节数。(本节稍后将对“非负”这一修饰词的某些例外进行说明。)通常,读、写操作都从当前文件位移量处开始,并使位移量增加所读或写的字节数。按系统默认,当
打开一个文件时,除非指定O A P P E N D选择项,否则该位移量被设置为0。
可以调用l s e e k显式地定位一个打开文件。
#include <sys/types.h>
#include <unistd.h>
off_t lseek(intf i l e d e s, off_t o f f s e t, int w h e n c e) ;
返回:若成功为新的文件位移,若出错为- 1
对参数offset 的解释与参数w h e n c e的值有关。
• 若w h e n c e是S E E K S E T,则将该文件的位移量设置为距文件开始处offset 个字节。
• 若w h e n c e是S E E K C U R,则将该文件的位移量设置为其当前值加offset, offset可为正或负。
• 若w h e n c e是S E E K E N D,则将该文件的位移量设置为文件长度加offset, offset可为正或负。
若l s e e k成功执行,则返回新的文件位移量,为此可以用下列方式确定一个打开文件的当前位移量:
off_t currpos;
currpos = lseek(fd, 0, SEEK_CUR);
这种方法也可用来确定所涉及的文件是否可以设置位移量。如果文件描述符引用的是一个管道或F I F O,则l s e e k返回-1,并将e r r n o设置为E P I P E。三个符号常数S E E K S E T,S E E K C U R和S E E K E N D是由系统V引进的。在系统V之前, w h e n c e被指定为0 (绝对位移量),1 ( 相对于当前位置的位移量)或2 (相对文件尾端的位移量)。很多软件仍直接使用这些数字进行编码。在l s e e k中的字符l表示长整型。在引入o ff t数据类型之前, o f f s e t参数和返回值是长整型的。l s e e k是由V 7引进的,当时C语言中增加了长整型。(在V 6中,用函数s e e k和t e l l提供类似功能。)

 

实例
程序3 - 1用于测试其标准输入能否被设置位移量。
程序3-1 测试标准输入能否被设置位移量
#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);
}

 

如果用交互方式调用此程序,则可得:
$ a.out < /etc/motd
seek OK
$ cat < /etc/mot|da . o u t
cannot seek
$ a.out < /var/spool/cron/FIFO
cannot seek
      通常,文件的当前位移量应当是一个非负整数,但是,某些设备也可能允许负的位移量。但对于普通文件,则其位移量必须是非负值。因为位移量可能是负值,所以在比较l s e e k的返回值时应当谨慎,不要测试它是否小于0,而要测试它是否等于-1。

     l s e e k仅将当前的文件位移量记录在内核内,它并不引起任何I / O操作。然后,该位移量用于下一个读或写操作。文件位移量可以大于文件的当前长度,在这种情况下,对该文件的下一次写将延长该文件,并在文件中构成一个空调,这一点是允许的。位于文件中但没有写过的字节都被读为0。


实例
程序3 - 2用于创建一个具有空洞的文件。
程序3-2 创建一个具有空洞的文件

#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 now = 16384 */

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

    exit(0);
}

运行该程序得到:
$ a . o u t
$ ls -1 file.hole 检查其大小
-rw-r--r-- 1 stevens 50 Jul 31 05:50 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
0000040 /0 /0 /0 /0 /0 /0 /0 /0 A B C D E F G H
0000060 I J
0 0 0 0 0 6 2
使用od ( 1 )命令观察该文件的实际内容。命令行中的- c标志表示以字符方式打印文件内容。从中可以看到,文件中间的3 0个未写字节都被读成为0。每一行开始的一个七位数是以八进制形式表示的字节位移量。本例调用了将在3 . 8节中说明的w r i t e函数。4 . 1 2节将对具有空洞的文件进行更多说明。

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值