嵌入式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;          // 纳秒部分设置为0
    new_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()调用失败,程序将打印错误消息并退出。运行结果如下:

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/695833.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

如何从SD存储卡恢复已删除的图像

SD卡概述 在日常工作和学习中&#xff0c;SD卡经常被用作许多设备上的存储设备&#xff0c;例如数码相机&#xff0c;手机&#xff0c;摄像机&#xff0c;行车记录仪以及监控设备&#xff0c;用于为我们存储图像&#xff0c;视频&#xff0c;文档&#xff0c;音频和其他数据。…

html5实现个人网站源码

文章目录 1.设计来源1.1 网站首页页面1.2 个人工具页面1.3 个人日志页面1.4 个人相册页面1.5 给我留言页面 2.效果和源码2.1 动态效果2.2 目录结构 源码下载 作者&#xff1a;xcLeigh 文章地址&#xff1a;https://blog.csdn.net/weixin_43151418/article/details/139564407 ht…

12. Django 第三方功能应用

12. 第三方功能应用 因为Django具有很强的可扩展性, 所以延伸了第三方功能应用. 通过本章的学习, 读者能够在网站开发过程中快速实现API接口开发, 验证码生成与使用, 站内搜索引擎, 第三方网站实现用户注册, 异步任务和定时任务, 即时通信等功能.12.1 Django Rest Framework框…

【YOLOv5进阶】——修改网络结构(以C2f模块为例)

一、站在巨人的肩膀上 这里我们借鉴YOLOv8源码&#xff1a; 上期说到&#xff0c;对于网络模块定义详情在common.py这个文件&#xff0c;如Conv、CrossConv、C3f等。本期要修改的需要参考YOLOv8里的C2f模块&#xff0c;它定义在YOLOv8的module文件夹的block.py文件里&#xf…

js调试过程中修改变量值

1.在想要变更的地方添加断点 2.添加监视表达式 3.执行网页代码&#xff0c;当执行到断点处则会停止 4.点击执行下一步&#xff0c;则会执行监视表达式

新材料正不断推动模具3D打印行业发展

随着工业4.0的浪潮席卷全球&#xff0c;模具制造行业也迎来了技术革新的新纪元。3D打印技术以其独特的制造优势&#xff0c;正逐渐在模具制造领域崭露头角。然而&#xff0c;要实现模具3D打印技术的广泛应用&#xff0c;高性能的打印材料是不可或缺的关键因素。 材料是模具3D打…

一、Socket创建和连接

C网络编程&#xff08;asio&#xff09; 文章目录 C网络编程&#xff08;asio&#xff09;1、Asio概述2、网络编程基本流程2、创建socket3、创建监听socket4、绑定accpet监听套接字5、连接指定的端点6、服务器接收连接 点击查看代码 1、Asio概述 ​ Asio起源于Boost库&#xf…

超详解——python条件和循环——小白篇

目录 1. 缩进和悬挂else 2. 条件表达式 3. 和循环搭配的else 4. 可调用对象 总结&#xff1a; 1. 缩进和悬挂else 在Python中&#xff0c;代码块是通过缩进来表示的。条件判断和循环结构的代码块需要正确缩进。悬挂else指的是else子句和相应的if或循环在同一级别的缩进。 …

AVL树 ---(C++)

本篇讲全面的讲解 AVL 树的插入&#xff0c;旋转以及验证 AVL 树的性能&#xff08;本篇未实现删除代码&#xff09;。至于为什么会有 AVL 树&#xff0c;这是因为简单的二叉搜索树并不能直接的保证搜索的效率&#xff0c;因为当我们在二叉搜索树中插入一段有序的序列的时候&am…

STC90C51驱动LCD1602、LCD12864、OLED

主控芯片&#xff08;STC90C516RDPG5151028&#xff09;介绍 ROM64K,RAM1280字节&#xff0c;40Pin&#xff0c;3个定时器&#xff0c;1个串口&#xff0c;8个中断源&#xff08;分别是&#xff1a;外部中断0(INTO)、外部中断 1(INT1)、外部中断 2(INT2)、外部中断 3(INT3)、定…

【微信小程序开发(从零到一)】——个人中心页面的实战项目(二)

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;开发者-曼亿点 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 曼亿点 原创 &#x1f468;‍&#x1f4bb; 收录于专栏&#xff1a…

「动态规划」如何计算能获得多少点数?

740. 删除并获得点数https://leetcode.cn/problems/delete-and-earn/description/ 给你一个整数数组nums&#xff0c;你可以对它进行一些操作。每次操作中&#xff0c;选择任意一个nums[i]&#xff0c;删除它并获得nums[i]的点数。之后&#xff0c;你必须删除所有等于nums[i] …

【网络安全的神秘世界】web应用程序安全与风险

&#x1f31d;博客主页&#xff1a;泥菩萨 &#x1f496;专栏&#xff1a;Linux探索之旅 | 网络安全的神秘世界 | 专接本 第一章&#xff1a;web应用程序安全与风险 web攻击基础知识 1、什么是web应用攻击 web攻击的本质&#xff0c;就是通过http协议篡改应用程序&#xff0…

虚拟机ping不通主机,但是主机可以ping通虚拟机

我在Windows10系统安装了虚拟机&#xff0c;设置的主机与虚拟机的连接方式是桥接&#xff0c;安装好后&#xff0c;发现虚拟机ping不通主机&#xff0c;但是主机可以ping通虚拟机。 我的操作是&#xff1a;关闭防火墙&#xff0c;发现虚拟机可以ping通主机了。说明是Windows10…

python后端结合uniapp与uview组件tabs,实现自定义导航按钮与小标签颜色控制

实现效果&#xff08;红框内&#xff09;&#xff1a; 后端api如下&#xff1a; task_api.route(/user/task/states_list, methods[POST, GET]) visitor_token_required def task_states(user):name_list [待接单, 设计中, 交付中, 已完成, 全部]data []color [#F04864, …

CPP初级:模板的运用!

目录 一.泛型编程 二.函数模板 1.函数模板概念 2.函数模板格式 3.函数模板的原理 三.函数模板的实例化 1.隐式实例化 2.显式实例化 3.模板参数的匹配原则 四.类模板 1.类模板的定义格式 2.类模板的实例化 一.泛型编程 泛型编程&#xff1a;编写与类型无关的通用代码…

express入门01服务器搭建以及get和post请求的监听

微搭提供了后端API的能力&#xff0c;但是不同的版本收费差别巨大&#xff0c;因为使用的门槛限制了中小企业使用低代码平台。那可不可以既要又要呢&#xff1f;答案是肯定的&#xff0c;那其实掌握一定的后端框架&#xff0c;借助我们在低代码中已经熟练掌握的技能其实是比较容…

2024.6.9 七

Python的time库 先导入库 import time相关函数 time.time() 返回当前时间的时间戳(一个记录时间的浮点数),从1970年开始算的 time.localtime(sec) 返回一个指定时间戳(sec)的struct_time对象,是一个元组封装起来的,默认是当地时间 struct_time对象 tm_year 年 tm_mon 月 tm_…

CDR2024软件破解Keygen激活工具2024最新版

CorelDRAW Graphics Suite2024最新版&#xff0c;这是一款让我爱不释手的图形设计神器&#xff01;作为一个软件评测专家&#xff0c;我一直在寻找一款能够提升我的设计效率和创造力的工具。而这款软件&#xff0c;简直就是为我量身定制的&#xff01;&#x1f389; 「CorelDR…

算法金 | AI 基石,无处不在的朴素贝叶斯算法

大侠幸会&#xff0c;在下全网同名「算法金」 0 基础转 AI 上岸&#xff0c;多个算法赛 Top 「日更万日&#xff0c;让更多人享受智能乐趣」 历史上&#xff0c;许多杰出人才在他们有生之年默默无闻&#xff0c; 却在逝世后被人们广泛追忆和崇拜。 18世纪的数学家托马斯贝叶斯…