简介
truncate()和ftruncate()系统调用将文件大小设置为length参数指定的值。
NAME
truncate, ftruncate - truncate a file to a specified length
SYNOPSIS
#include <unistd.h>
#include <sys/types.h>
int truncate(const char *path, off_t length);
int ftruncate(int fd, off_t length);
若文件当前长度大于参数length,调用将丢弃超出部分,若小于参数length,调用将在文件尾部添加一系列空字节或是一个文件空洞。
两个系统调用之间的差别在于如何指定操作文件。truncate()以路径名字符串来指定文件,并要求可访问该文件,且对文件拥有写权限。若文件名为符号链接,那么调用将对其进行解引用。而调用ftruncate()之前,需以可写方式打开操作文件,获取其文件描述符以指代该文件,该系统调用不会修改文件偏移量。
若ftruncate()的length参数值超出文件的当前大小,SUSv3允许两种行为:要么扩展该文件(如Linux),要么返回错误。而符合XSI标准的系统则必须采取前一种行为。相同的情况,对于truncate()系统调用,SUSv3则要求总是能扩展文件。
truncate()无需先以open()(或是一些其他方法)来获取文件描述符,却可修改文件内容,在系统调用中可谓独树一帜。
创建两个测试文件file1.txt file2.txt
echo "hello world" > file1.txt
echo "hello world" > file2.txt
测试一 :truncate例子
#ifndef _GNU_SOURCE
#define _GNU_SOURCE
#endif
#include <sys/types.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <getopt.h>
#include <sys/epoll.h>
#include <sys/uio.h>
#include <limits.h>
#define _DEBUG_INFO
#ifdef _DEBUG_INFO
#define DEBUG_INFO(format, ...) printf("%s:%d $$ " format "\n" \
,__func__,__LINE__ \
, ##__VA_ARGS__)
#else
#define DEBUG_INFO(format, ...)
#endif
static char *file_name = "writev.txt";
int main(int argc, char **argv)
{
int fd;
truncate("file1.txt",6);
truncate("file2.txt",100);
return 0;
}
执行测试程序,执行后,文件长度分别变成了6和100.
重新创建这两个文件
rm file*
创建两个测试文件file1.txt file2.txt
echo "hello world" > file1.txt
echo "hello world" > file2.txt
测试二 :ftruncate例子
#ifndef _GNU_SOURCE
#define _GNU_SOURCE
#endif
#include <sys/types.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <getopt.h>
#include <sys/epoll.h>
#include <sys/uio.h>
#include <limits.h>
#define _DEBUG_INFO
#ifdef _DEBUG_INFO
#define DEBUG_INFO(format, ...) printf("%s:%d $$ " format "\n" \
,__func__,__LINE__ \
, ##__VA_ARGS__)
#else
#define DEBUG_INFO(format, ...)
#endif
static char *file_name = "writev.txt";
int main(int argc, char **argv)
{
int fd1 = open("file1.txt", O_WRONLY);
int fd2 = open("file2.txt", O_WRONLY);
if(fd1 < 0 || fd2 < 0){
perror("open");
return -1;
}
int res1 = ftruncate(fd1,6);
int res2 = ftruncate(fd2,100);
if(res1 < 0 || res2 < 0){
perror("ftruncate");
return -1;
}
DEBUG_INFO("fd1 offset = %d",lseek(fd1,0,SEEK_CUR));
DEBUG_INFO("fd2 offset = %d",lseek(fd2,0,SEEK_CUR));
close(fd1);
close(fd2);
return 0;
}
执行之前,文件长度都是12
执行后,长度变为6和100
输出信息:表明 ftruncate不会改变文件位置。
测试三,LFS大文件验证
创建一个8GB的空洞文件。
#ifndef _GNU_SOURCE
#define _GNU_SOURCE
#endif
#include <sys/types.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <getopt.h>
#include <sys/epoll.h>
#include <sys/uio.h>
#include <limits.h>
#define _DEBUG_INFO
#ifdef _DEBUG_INFO
#define DEBUG_INFO(format, ...) printf("%s:%d $$ " format "\n" \
,__func__,__LINE__ \
, ##__VA_ARGS__)
#else
#define DEBUG_INFO(format, ...)
#endif
static char *file_name = "writev.txt";
int main(int argc, char **argv)
{
int fd1 = open("file1.txt", O_WRONLY);
int fd2 = open("file2.txt", O_WRONLY);
if(fd1 < 0 || fd2 < 0){
perror("open");
return -1;
}
long long_len_1 = 1*(long)1024*(long)1024*(long)1024;
long long_len_2 = 8*(long)1024*(long)1024*(long)1024;
DEBUG_INFO("%ld,%x",long_len_1,long_len_1);
DEBUG_INFO("%ld,%x",long_len_2,long_len_2);
DEBUG_INFO("%ld,%ld",sizeof(long_len_1),sizeof(off_t));
int res1 = ftruncate(fd1,long_len_1);
int res2 = ftruncate(fd2,long_len_2);
if(res1 < 0 || res2 < 0){
perror("ftruncate");
return -1;
}
DEBUG_INFO("fd1 offset = %ld",lseek(fd1,0,SEEK_CUR));
DEBUG_INFO("fd2 offset = %ld",lseek(fd2,0,SEEK_CUR));
close(fd1);
close(fd2);
return 0;
}
执行后,如下所示:看到一个超级大的8GB的文件。
$ ls -lsh file*
0 -rw-rw-r-- 1 lkmao lkmao 1.0G 6月 28 21:08 file1.txt
4.0K -rw-rw-r-- 1 lkmao lkmao 8.0G 6月 28 21:08 file2.txt