嵌入式Linux系统编程 — 3.5 utime、utimes、futimens、utimensat函数修改文件时间属性

本文主要是介绍嵌入式Linux系统编程 — 3.5 utime、utimes、futimens、utimensat函数修改文件时间属性,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

1 文件的时间属性简介

2 utime()函数

2.1 utime()函数简介

2.2 示例程序

3 utimes()函数

3.1 utimes()函数简介

3.2 示例程序 

4  futimens()函数

4.1 futimens()函数简介

4.2 示例程序

5 utimensat()函数

5.1 utimensat()函数简介

5.2 示例程序


1 文件的时间属性简介

在Linux系统中,文件的时间属性通常与文件的状态信息一起存储在文件的inode中。在<sys/stat.h>头文件中,结构体struct stat用于存储文件的状态信息,包括时间属性。以下是struct stat中与时间属性相关的成员变量:

st_atime文件的最后访问时间(Access Time),访问指的是读取文件内容,文件内容最后一次被读取的时间。
st_mtime表示文件的最后修改时间(Modification Time),文件内容发生改变,譬如使用 write()函数写入数据到文件中。
st_ctime文件的最后状态改变时间(Status Change Time),状态更改指的是该文件的 inode 节点最后一次被修改的时间,譬如更改文件的访问权限、更改文件的用户 ID、用户组 ID、更改链接数等。

下面表列出了一些系统调用或 C 库函数对文件时间属性的影响,有些操作并不仅仅只会影响文件本身的时间属性,还会影响到其父目录的相关时间属性。

不同函数对文件时间属性的影响

2 utime()函数

2.1 utime()函数简介

utime()函数在Linux系统中用于设置文件的访问时间(access time)和修改时间(modification time)。这个函数允许你指定文件的访问时间和修改时间,而不是依赖系统自动更新的时间。函数原型如下:

#include <utime.h> int utime(const char *filename, const struct utimbuf *times);
  • filename:指定要修改时间属性的文件的路径。
  • times:指向struct utimbuf结构的指针,该结构包含了要设置的访问时间和修改时间。如果timesNULL,则系统将当前时间作为访问时间和修改时间。

struct utimbuf定义如下:

struct utimbuf { time_t actime; // 访问时间 time_t modtime; // 修改时间 
};

该结构体中包含了两个 time_t 类型的成员,分别用于表示访问时间和内容修改时间, time_t 类型其实就是 long int 类型,所以这两个时间是以秒为单位的,所以由此可知, utime()函数设置文件的时间属性精度只能到秒。

同样对于文件来说,时间属性也是文件非常重要的属性之一,对文件时间属性的修改也不是任何用户都可以随便修改的, 只有以下两种进程可对其进行修改:

  • 超级用户进程(以 root 身份运行的进程) 。
  • 有效用户 ID 与该文件用户 ID(文件所有者)相匹配的进程。
  • 在参数 times 等于 NULL 的情况下,对文件拥有写权限的进程。

2.2 示例程序

下面的示例程序,接受一个命令行参数作为文件名,并尝试更新这个文件的时间属性:

#include <stdio.h>
#include <stdlib.h>
#include <utime.h>
#include <time.h>
#include <string.h>int main(int argc, char *argv[]) {// 检查命令行参数数量if (argc != 2) {fprintf(stderr, "Usage: %s <filename>\n", argv[0]);exit(EXIT_FAILURE);}// 获取命令行参数中的文件名const char *filename = argv[1];// 获取当前时间time_t current_time = time(NULL);// 创建utimbuf结构体并设置时间struct utimbuf new_times;new_times.actime = new_times.modtime = current_time;// 使用utime()函数更新文件时间if (utime(filename, &new_times) == -1) {// 如果utime()调用失败,打印错误信息perror("Error updating file times");exit(EXIT_FAILURE);}printf("File '%s' access and modification times have been updated to: %s", filename, ctime(&current_time));return 0;
}

程序首先检查命令行参数的数量,如果参数数量不正确,程序将打印正确的用法并退出。进一步获取当前时间,设置utimbuf结构体,并调用utime()函数来更新文件的时间属性。如果utime()调用失败,程序将使用perror()打印错误消息并退出。如果调用成功,程序将打印一条消息,告知用户文件的时间属性已被更新。运行结果如下:

utimes()函数

3.1 utimes()函数简介

utimes()函数用于设置文件的访问时间(access time)和修改时间(modification time)的函数。与utime()函数不同,utimes()允许你指定更精确的时间,包括纳秒级别的精度。函数原型如下:

#include <sys/time.h> int utimes(const char *filename, const struct timeval times[2]);
  • filename:指定要修改时间属性的文件的路径。
  • times:指向包含两个struct timeval结构的数组的指针。第一个struct timeval用于设置访问时间,第二个用于设置修改时间。如果timesNULL,则系统将当前时间作为访问时间和修改时间。

struct timeval定义如下:

struct timeval { time_t tv_sec; // 时间的秒部分 suseconds_t tv_usec; // 时间的微秒部分 
};

3.2 示例程序 

下面是一个使用utimes()函数的示例程序:

#include <stdio.h>
#include <stdlib.h>
#include <sys/time.h>
#include <unistd.h>
#include <string.h>int main(int argc, char *argv[]) {if (argc != 3) {fprintf(stderr, "Usage: %s <filename> <time>\n", argv[0]);exit(EXIT_FAILURE);}const char *filename = argv[1];const char *time_str = argv[2];// 将字符串时间转换为timeval结构struct timeval new_times[2];if (sscanf(time_str, "%ld.%ld", &new_times[0].tv_sec, &new_times[0].tv_usec) != 2) {fprintf(stderr, "Invalid time format.\n");exit(EXIT_FAILURE);}new_times[1] = new_times[0]; // 访问时间和修改时间相同// 使用utimes()函数更新文件时间if (utimes(filename, new_times) == -1) {perror("Error updating file times");exit(EXIT_FAILURE);}printf("File '%s' access and modification times have been updated.\n", filename);return 0;
}

示例程序首先检查命令行参数的数量,确保用户提供了文件名和时间字符串。然后将时间字符串转换为timeval结构,这里假设时间字符串的格式为秒和微秒的组合,例如"1234567890.987654"。接着,我们使用utimes()函数来更新文件的访问时间和修改时间。如果转换失败或utimes()调用失败,程序将打印错误消息并退出。运行结果如下:

4  futimens()函数

除了上面给大家介绍了两个系统调用外,futimens()和 utimensat()函数功能与 utime()和 utimes()函数功能一样,用于显式修改文件时间戳,这两个系统调用相对于 utime 和 utimes 函数有以下三个优点:

  • 高精度时间设置futimens()utimensat()允许以纳秒为单位设置文件的时间戳,而utimes()只提供微秒级的精度,这是对时间设置精度的一个显著提升。

  • 单独设置时间戳:这两个系统调用可以独立地设置访问时间或修改时间,用户可以只更改其中一个时间戳而保持另一个不变。使用utime()utimes()时,如果要单独设置一个时间戳,需要先使用stat()获取另一个时间戳的当前值,然后再进行设置。

  • 灵活设置当前时间futimens()utimensat()可以单独将任一时间戳设置为当前时间,而使用utime()utimes()时,如果将times参数设置为NULL,则会将所有时间戳都设置为当前时间,无法只更新其中一个。

4.1 futimens()函数简介

futimens()函数是用于设置文件的时间属性,特别是针对已经打开的文件描述符。这个函数允许你为文件的访问时间(access time)和修改时间(modification time)设置精确到纳秒的时间戳。函数原型如下:

#include <sys/stat.h> int futimens(int fd, const struct timespec times[2]);
  • fd:文件描述符,是一个整数值,表示要设置时间属性的打开文件。
  • times:指向包含两个struct timespec结构的数组的指针。第一个struct timespec用于设置访问时间,第二个用于设置修改时间。如果timesNULL,则系统将当前时间作为访问时间和修改时间。

struct timespec定义如下:

struct timespec { time_t tv_sec; // 时间的秒部分 long tv_nsec; // 时间的纳秒部分 
};

4.2 示例程序

下面是一个使用futimens()函数的示例程序:

#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>
#include <time.h>  // 包含time.h头文件int main(int argc, char *argv[]) {if (argc != 2) {fprintf(stderr, "Usage: %s <filename>\n", argv[0]);exit(EXIT_FAILURE);}const char *filename = argv[1];int fd;// 打开文件以获取文件描述符fd = open(filename, O_RDONLY);if (fd == -1) {perror("Error opening file");exit(EXIT_FAILURE);}// 创建timespec结构体并设置当前时间struct timespec new_times[2];new_times[0].tv_sec = time(NULL);  // 获取当前时间的秒数new_times[0].tv_nsec = 0;          // 纳秒部分设置为0new_times[1] = new_times[0];       // 修改时间和访问时间设置为相同// 使用futimens()函数更新文件时间if (futimens(fd, new_times) == -1) {perror("Error updating file times");close(fd);exit(EXIT_FAILURE);}printf("File '%s' access and modification times have been updated.\n", filename);// 关闭文件描述符close(fd);return 0;
}

程序首先检查命令行参数的数量,确保用户提供了文件名。然后,我们使用open()函数以只读模式打开文件,并获取文件描述符fd。接着创建了两个timespec结构体实例,并将它们设置为当前时间。然后调用futimens()函数来更新文件的访问时间和修改时间。程序运行结果如下:

5 utimensat()函数

5.1 utimensat()函数简介

utimensat()函数允许以纳秒级的精度来设置文件或目录的时间戳。这个函数提供了比传统的utime()utimes()函数更高的时间设置精度,并且具有更多的灵活性。函数原型如下:

#include <sys/stat.h> int utimensat(int dirfd, const char *pathname, const struct timespec times[2], int flags);
  • dirfd:该参数可以是一个目录的文件描述符,也可以是特殊值 AT_FDCWD;如果 pathname 参数指定的是文件的绝对路径,则此参数会被忽略。
  • pathname:这是要修改时间戳的文件或目录的路径。
  • times:这是一个指向struct timespec数组的指针,包含两个时间戳,分别用于设置访问时间(times[0])和修改时间(times[1])。如果timesNULL,则时间戳将被设置为当前时间。
  • flags:这是一些标志位,可以是以下值的组合:
    • AT_SYMLINK_NOFOLLOW:如果pathname是一个符号链接,则utimensat()将修改链接指向的目标而不是链接本身。

5.2 示例程序

下面的使用utimensat()函数示例程序接受一个文件名和一个时间字符串作为命令行参数,并尝试将该文件的访问和修改时间设置为指定的时间。时间字符串的格式应该是秒.纳秒

#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>int main(int argc, char *argv[]) {if (argc != 3) {fprintf(stderr, "Usage: %s <filename> <timestamp>\n", argv[0]);fprintf(stderr, "Timestamp format: seconds.nanoseconds\n");exit(EXIT_FAILURE);}const char *filename = argv[1];const char *timestamp_str = argv[2];char *dot;unsigned long seconds, nanoseconds;// 解析时间字符串seconds = strtoul(timestamp_str, &dot, 10);if (*dot != '.') {fprintf(stderr, "Invalid timestamp format.\n");exit(EXIT_FAILURE);}// 跳过点号,解析纳秒部分nanoseconds = strtoul(dot + 1, NULL, 10);// 创建timespec结构体struct timespec new_times[2];new_times[0].tv_sec = seconds;  // 访问时间new_times[0].tv_nsec = nanoseconds;  // 访问时间的纳秒部分new_times[1].tv_sec = seconds;  // 修改时间new_times[1].tv_nsec = nanoseconds;  // 修改时间的纳秒部分// 使用utimensat()函数更新文件时间if (utimensat(AT_FDCWD, filename, new_times, 0) == -1) {perror("Error updating file times");exit(EXIT_FAILURE);}printf("File '%s' access and modification times have been updated to: %lu.%09lu\n",filename, seconds, nanoseconds);return 0;
}

程序首先检查命令行参数的数量是否正确。接,我们解析时间字符串以获取秒和纳秒部分,然后创建timespec结构体并设置相应的时间。最后使用utimensat()函数尝试更新文件的访问和修改时间。如果utimensat()调用失败,程序将打印错误消息并退出。运行结果如下:

 

这篇关于嵌入式Linux系统编程 — 3.5 utime、utimes、futimens、utimensat函数修改文件时间属性的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1048687

相关文章

linux生产者,消费者问题

pthread_cond_wait() :用于阻塞当前线程,等待别的线程使用pthread_cond_signal()或pthread_cond_broadcast来唤醒它。 pthread_cond_wait() 必须与pthread_mutex 配套使用。pthread_cond_wait()函数一进入wait状态就会自动release mutex。当其他线程通过pthread

问题:第一次世界大战的起止时间是 #其他#学习方法#微信

问题:第一次世界大战的起止时间是 A.1913 ~1918 年 B.1913 ~1918 年 C.1914 ~1918 年 D.1914 ~1919 年 参考答案如图所示

Linux 安装、配置Tomcat 的HTTPS

Linux 安装 、配置Tomcat的HTTPS 安装Tomcat 这里选择的是 tomcat 10.X ,需要Java 11及更高版本 Binary Distributions ->Core->选择 tar.gz包 下载、上传到内网服务器 /opt 目录tar -xzf 解压将解压的根目录改名为 tomat-10 并移动到 /opt 下, 形成个人习惯的路径 /opt/tomcat-10

RedHat运维-Linux文本操作基础-AWK进阶

你不用整理,跟着敲一遍,有个印象,然后把它保存到本地,以后要用再去看,如果有了新东西,你自个再添加。这是我参考牛客上的shell编程专项题,只不过换成了问答的方式而已。不用背,就算是我自己亲自敲,我现在好多也记不住。 1. 输出nowcoder.txt文件第5行的内容 2. 输出nowcoder.txt文件第6行的内容 3. 输出nowcoder.txt文件第7行的内容 4. 输出nowcode

【Linux进阶】UNIX体系结构分解——操作系统,内核,shell

1.什么是操作系统? 从严格意义上说,可将操作系统定义为一种软件,它控制计算机硬件资源,提供程序运行环境。我们通常将这种软件称为内核(kerel),因为它相对较小,而且位于环境的核心。  从广义上说,操作系统包括了内核和一些其他软件,这些软件使得计算机能够发挥作用,并使计算机具有自己的特生。这里所说的其他软件包括系统实用程序(system utility)、应用程序、shell以及公用函数库等

零基础STM32单片机编程入门(一)初识STM32单片机

文章目录 一.概要二.单片机型号命名规则三.STM32F103系统架构四.STM32F103C8T6单片机启动流程五.STM32F103C8T6单片机主要外设资源六.编程过程中芯片数据手册的作用1.单片机外设资源情况2.STM32单片机内部框图3.STM32单片机管脚图4.STM32单片机每个管脚可配功能5.单片机功耗数据6.FALSH编程时间,擦写次数7.I/O高低电平电压表格8.外设接口

16.Spring前世今生与Spring编程思想

1.1.课程目标 1、通过对本章内容的学习,可以掌握Spring的基本架构及各子模块之间的依赖关系。 2、 了解Spring的发展历史,启发思维。 3、 对 Spring形成一个整体的认识,为之后的深入学习做铺垫。 4、 通过对本章内容的学习,可以了解Spring版本升级的规律,从而应用到自己的系统升级版本命名。 5、Spring编程思想总结。 1.2.内容定位 Spring使用经验

【操作系统】信号Signal超详解|捕捉函数

🔥博客主页: 我要成为C++领域大神🎥系列专栏:【C++核心编程】 【计算机网络】 【Linux编程】 【操作系统】 ❤️感谢大家点赞👍收藏⭐评论✍️ 本博客致力于知识分享,与更多的人进行学习交流 ​ 如何触发信号 信号是Linux下的经典技术,一般操作系统利用信号杀死违规进程,典型进程干预手段,信号除了杀死进程外也可以挂起进程 kill -l 查看系统支持的信号

通信系统网络架构_2.广域网网络架构

1.概述          通俗来讲,广域网是将分布于相比局域网络更广区域的计算机设备联接起来的网络。广域网由通信子网于资源子网组成。通信子网可以利用公用分组交换网、卫星通信网和无线分组交换网构建,将分布在不同地区的局域网或计算机系统互连起来,实现资源子网的共享。 2.网络组成          广域网属于多级网络,通常由骨干网、分布网、接入网组成。在网络规模较小时,可仅由骨干网和接入网组成

Windows/macOS/Linux 安装 Redis 和 Redis Desktop Manager 可视化工具

本文所有安装都在macOS High Sierra 10.13.4进行,Windows安装相对容易些,Linux安装与macOS类似,文中会做区分讲解 1. Redis安装 1.下载Redis https://redis.io/download 把下载的源码更名为redis-4.0.9-source,我喜欢跟maven、Tomcat放在一起,就放到/Users/zhan/Documents