编写一个程序,使用O_APPEND标志并以写方式打开一个已存在的文件,且将文件偏移量置于文件起始处,再写入数据,数据会显示在文件的哪个位置?为什么?
该练习为5-2,5-1由于系统本身为64位,在修改api后以及屏蔽新增宏一样可以生成大文件,就不做了:
接下来是习题5-2,首先创建一个测试文件test:
然后编写o_appendtest.c,首先是不加标志位O_APPEND :
/*##############################################
#
# Author: Lizhenggen - 762741539@qq.com
#
# QQ : 762741539
#
# Last modified: 2022-04-17 09:00
#
# Filename: large_file.c
##############################################*/
#define _LARGEFILE64_SOURCE
#include <sys/stat.h>
#include <fcntl.h>
#include "tlpi_hdr.h"
int main(int argc, char *argv[])
{
int fd;
off64_t off;
if (argc != 2 || strcmp(argv[1], "--help") == 0)
usageErr("%s pathname\n", argv[0]);
fd = open(argv[1], O_WRONLY);//以只写的方式打开一个已经存在的文件
if (fd == -1)
errExit("open64");
if (lseek64(fd, 0, SEEK_SET) == -1) //将文件指针偏移值放置于起始处
errExit("lseek64");
if (write(fd, "test", 4) == -1) //在起始处写入“test”这四个字节
errExit("write");
exit(EXIT_SUCCESS);
}
编写Makefile:
LIBINCL = -I/apue/tlpi-book/lib
CPUBLIC = /apue/tlpi-book/lib/get_num.c /apue/tlpi-book/lib/error_functions.c
CFLAGS = -g
all : copy kongdong large_file o_appendtest
copy : copy.c
gcc $(CFLAGS) -o copy copy.c $(LIBINCL) $(CPUBLIC) -lm -lc
kongdong : kongdong.c
gcc $(CFLAGS) -o kongdong kongdong.c $(LIBINCL) $(CPUBLIC) -lm -lc
large_file : large_file.c
gcc $(CFLAGS) -o large_file large_file.c $(LIBINCL) $(CPUBLIC) -lm -lc
o_appendtest : o_appendtest.c
gcc $(CFLAGS) -o o_appendtest o_appendtest.c $(LIBINCL) $(CPUBLIC) -lm -lc
clean:
rm copy kongdong large_file o_appendtest
编译相关文件,并运行o_appendtest,将数据写入到test文件中:
查看test中的数据,可以看到hello为test所覆盖,注意这里是未加标识符O_APPEND的测试结果:
接下来再o_appendtest.c中加入标志位O_APPEND:
编译运行后,查看test中的结果,可以看到它是在文件的末端增加新增字符:
这是由于O_APPEND打开后,是一个原子操作:移动到末端,写数据。这是O_APPEND打开的作用。中间的插入时无效的。同时代码中的lseek函数是没有用的,write的写入是到末端。在write写完后,标志位是在文件末端,这个时候使用read函数将读不到任何数据。
当然这里出现了换行,具体原因未知,待后续。