前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >11.8 实现重置文件时间戳

11.8 实现重置文件时间戳

作者头像
微软技术分享
发布2023-11-21 09:03:49
1770
发布2023-11-21 09:03:49
举报

11.8.1 切割文件名与路径

如下代码是一段文件路径切割实现,通过传入文件路径,获取文件名和文件路径的功能。具体实现包括两个函数:GetFileNameGetFilePath。前者接收一个文件路径字符串,并返回该文件路径中的文件名;后者接收一个文件路径字符串,并返回该文件路径中除文件名以外的部分,即文件路径。

main函数中,首先定义了两个文件路径字符串szPathAszPathB。然后,分别调用GetFileNameGetFilePath函数,将它们的返回值保存在指针变量ptr中,并输出到控制台上。最后,程序返回0,结束执行。

代码语言:javascript
复制
#include <iostream>
#include <Windows.h>
#include <string.h>

// 传入路径得到文件名
char* GetFileName(char* Path)
{
    if (strchr(Path, '\\'))
    {
        char ch = '\\';
        char* ref = strrchr(Path, ch) + 1;
        return ref;
    }
    else
    {
        char ch = '/';
        char* ref = strrchr(Path, ch) + 1;
        return ref;
    }
}

// 传入路径只得到文件路径
char* GetFilePath(char *Path)
{
    int i, pos;
    for (int count = 0; count < strlen(Path); count++)
    {
        if (Path[count] == '\\' || Path[count] == '/')
        {
            pos = count;
        }
    }
    Path[pos] = '\0';
    return Path;
}

int main(int argc,char *argv[])
{
    char szPathA[128] = "C://Windows/LanguageOverlayCache/server.cpp";
    char szPathB[128] = "c:\\Windows\\LanguageOverlayCache\\kernel.dll";
    char* ptr = NULL;

    ptr = GetFileName(szPathA);
    std::cout << "文件名A: " << ptr << std::endl;

    ptr = GetFileName(szPathB);
    std::cout << "文件名B: " << ptr << std::endl;

    ptr = GetFilePath(szPathB);
    std::cout << "文件路径: " << ptr << std::endl;

    return 0;
}

传入一个完整文件路径,并自动分割;

11.8.2 遍历目录下文件

如下代码是一个使用递归遍历目录,并输出指定格式的文件信息的程序。主要用到了文件操作函数findfirst()_findnext()_findclose(),以及结构体类型_finddata_t

findfirst函数是Windows平台上用于查找文件的函数之一,它属于 C Runtime Library(CRT)中的一部分,提供了一种在指定目录中搜索文件的机制。

定义:

代码语言:javascript
复制
intptr_t _findfirst(
    const char *filespec,
    struct _finddata_t *fileinfo
);

参数:

  • filespec:指定要搜索的文件规范(通配符模式),可以包含路径信息。
  • fileinfo:指向 finddata_t 结构的指针,用于存储找到的文件信息。

返回值:

  • 如果成功,返回一个查找句柄(handle);如果失败,返回 -1。

findnext函数是Windows平台上用于查找文件的函数之一,它通常与_findfirst配合使用,用于获取指定目录中的下一个文件。

定义:

代码语言:javascript
复制
int _findnext(
    intptr_t handle,
    struct _finddata_t *fileinfo
);

参数:

  • handle:由findfirst返回的查找句柄。
  • fileinfo:指向finddata_t结构的指针,用于存储找到的下一个文件的信息。

返回值:

  • 如果成功,返回 0;如果失败或到达目录尾部,返回 -1。

函数dfsFolder()分别接收目录路径和需要查找的文件格式。通过使用_findfirst()找到该路径下的第一个文件或文件夹,如果是文件夹,则递归调用dfsFolder()函数,如果是文件,则判断其是否为需要查找的格式,如果是则输出该文件的相关信息。

函数dfsFolderAll()只传入了一个参数,即目录路径。该函数使用了和dfsFolder()类似的方法,但是不判断文件格式,而是将该路径下的所有文件和文件夹都列出来,在main()函数中,可以通过调用这两个函数来实现列出目录下所有文件和文件夹,或是列出目录下所有指定格式的文件。

代码语言:javascript
复制
#include <io.h>
#include <iostream>

using namespace std;

// 列举出所有文件路径
void dfsFolderAll(string folderPath)
{
    _finddata_t file;
    int k;
    long HANDLE;
    k = HANDLE = _findfirst(folderPath.c_str(), &file);
    while (k != -1)
    {
        cout << file.name << endl;
        k = _findnext(HANDLE, &file);
    }
    _findclose(HANDLE);
}

// 传入目录与需要列出的指定格式文件
void dfsFolder(string folderPath,string subpath)
{
    _finddata_t FileInfo;
    /*  文件信息结构体
    struct _finddata_t{
         unsigned attrib;     // 文件属性
         time_t time_create;    // 创建时的时间戳
         time_t time_access;    // 最后一次被访问时的时间戳
         time_t time_write;     // 最后一次被修改时的时间戳
         _fsize_t size;       // 文件字节大小
         char name[_MAX_FNAME];   // 文件名
    };*/
    string strfind = folderPath + "\\*";
    long Handle = _findfirst(strfind.c_str(), &FileInfo);

    if (Handle == -1L)
    {
        exit(0);
    }
    do
    {
        // 判断是否为目录 
        if (FileInfo.attrib & _A_SUBDIR)
        {
            // 判断目录是否为当前目录和上一级目录
            if ((strcmp(FileInfo.name, ".") != 0) && (strcmp(FileInfo.name, "..") != 0))
            {
                // 如果不是则拼接当前路径继续递归调用
                string newPath = folderPath + "\\" + FileInfo.name;
                dfsFolder(newPath,subpath.c_str());
            }
        }
        else
        {
            // 判断是不是指定后缀的文件
            if (strstr(FileInfo.name, subpath.c_str()))
            {
                cout << " 文件名: " << FileInfo.name << " 文件大小: " << FileInfo.size;
        if (FileInfo.attrib == _A_NORMAL)
        {
                    cout << " 普通文件 " << endl;
        }
        else if (FileInfo.attrib == _A_RDONLY)
        {
          cout << " 只读文件 " << endl;
        }
        else if (FileInfo.attrib == _A_HIDDEN)
        {
          cout << " 隐藏文件 " << endl;
        }
        else if (FileInfo.attrib == _A_SYSTEM)
        {
          cout << " 系统文件 " << endl;
        }
        else if (FileInfo.attrib == _A_SUBDIR)
        {
          cout << " 子目录 " << endl;
        }
        else
        {
          cout << " 存档文件 " << endl;
        }
            }
        }
    } while (_findnext(Handle, &FileInfo) == 0);
    _findclose(Handle);
}

int main(int argc,char *argv[])
{
    //dfsFolder("C:\\Windows\\system32",".dll");
    dfsFolderAll("C:\\Windows\\system32\\*");

    return 0;
}

运行后输出所有C:\\Windows\\system32\\*目录下的文件;

11.8.3 重置文件目录时间戳

文件目录时间戳是指与文件或目录相关联的时间信息,通常包括三个主要的时间戳:

  • 创建时间(Creation Time): 表示文件或目录被创建的时间。这个时间戳记录了文件或目录在文件系统中第一次被创建的时间点。
  • 最后访问时间(Last Access Time): 表示文件或目录最后一次被访问的时间。每当文件或目录被打开、读取、执行等操作时,最后访问时间都会更新。
  • 最后修改时间(Last Write Time): 表示文件或目录最后一次被修改的时间。当文件内容发生变化、文件被写入时,最后修改时间会更新。

这些时间戳提供了关于文件或目录的重要信息,对于文件管理和调查文件活动非常有用。在Windows和许多其他操作系统中,这些时间戳通常以 FILETIME 结构体的形式存储,该结构体表示从1601年1月1日午夜开始计算的100纳秒间隔数。

这些时间戳可以通过文件系统或相关的系统调用函数(如 GetFileTime、SetFileTime)来访问和修改。

SystemTimeToFileTime 用于将SYSTEMTIME结构体表示的时间转换为FILETIME结构体表示的时间。

函数签名:

代码语言:javascript
复制
BOOL SystemTimeToFileTime(
  const SYSTEMTIME *lpSystemTime,
  LPFILETIME       lpFileTime
);

参数:

  • lpSystemTime:指向 SYSTEMTIME 结构体的指针,表示待转换的系统时间。
  • lpFileTime:指向 FILETIME 结构体的指针,用于存储转换后的文件时间。

返回值:

  • 如果函数成功,返回非零值;如果函数失败,返回零。可以通过调用 GetLastError 函数获取更多信息。

LocalFileTimeToFileTime 用于将本地时间(FILETIME 结构体表示)转换为协调世界时(UTC)时间(同样是 FILETIME 结构体表示)。

函数签名:

代码语言:javascript
复制
BOOL LocalFileTimeToFileTime(
  const FILETIME *lpLocalFileTime,
  LPFILETIME     lpFileTime
);

参数:

  • lpLocalFileTime:指向 FILETIME 结构体的指针,表示待转换的本地时间。
  • lpFileTime:指向 FILETIME 结构体的指针,用于存储转换后的 UTC 时间。

返回值:

  • 如果函数成功,返回非零值;如果函数失败,返回零。可以通过调用 GetLastError 函数获取更多信息。

SetFileTime 用于设置指定文件的创建时间、访问时间和修改时间。

函数签名:

代码语言:javascript
复制
BOOL SetFileTime(
  HANDLE      hFile,
  const FILETIME *lpCreationTime,
  const FILETIME *lpLastAccessTime,
  const FILETIME *lpLastWriteTime
);

参数:

  • hFile:要设置时间信息的文件的句柄。
  • lpCreationTime:指向 FILETIME 结构体的指针,表示文件的创建时间。
  • lpLastAccessTime:指向 FILETIME 结构体的指针,表示文件的最后访问时间。
  • lpLastWriteTime:指向 FILETIME 结构体的指针,表示文件的最后修改时间。

返回值:

  • 如果函数成功,返回非零值;如果函数失败,返回零。可以通过调用 GetLastError 函数获取更多信息。
代码语言:javascript
复制
#include <io.h>
#include <iostream>
#include <windows.h>

using namespace std;

// 修改文件当前创建日期
bool SetLocalFileTime(const char* FilePath, const char* Date, const char* Time)
{
    HANDLE hFile = CreateFileA(FilePath, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL);
    if (hFile == INVALID_HANDLE_VALUE)
    {
        return false;
    }

    SYSTEMTIME spec_time;
    sscanf(Date, "%d-%d-%d", &spec_time.wYear, &spec_time.wMonth, &spec_time.wDay);
    sscanf(Time, "%d:%d:%d", &spec_time.wHour, &spec_time.wMinute, &spec_time.wSecond);

    FILETIME ft, LocalFileTime;
    SystemTimeToFileTime(&spec_time, &ft);
    LocalFileTimeToFileTime(&ft, &LocalFileTime);

    if (SetFileTime(hFile,
        &LocalFileTime,    // 创建时间
        &LocalFileTime,    // 访问时间
        &LocalFileTime     // 修改时间
        ))
    {
        CloseHandle(hFile);
        return true;
    }
    CloseHandle(hFile);
    return false;
}

// 批量修改文件名
void BatchSetFileDateTime(string folderPath, string Date, string Time)
{
    _finddata_t FileInfo;
    string strfind = folderPath + "\\*";
    long Handle = _findfirst(strfind.c_str(), &FileInfo);

    if (Handle == NULL)
    {
        exit(0);
    }
    do
    {
        // 判断是否为目录 
        if (FileInfo.attrib & _A_SUBDIR)
        {
            // 判断目录是否为当前目录和上一级目录
            if ((strcmp(FileInfo.name, ".") != 0) && (strcmp(FileInfo.name, "..") != 0))
            {
                // 如果不是则拼接当前路径继续递归调用
                string newPath = folderPath + "\\" + FileInfo.name;
                bool ref = SetLocalFileTime(newPath.c_str(), Date.c_str(), Time.c_str());
                if (ref == true)
                {
                    std::cout << "[*] 目录: " << newPath.c_str() << std::endl;
                }
                BatchSetFileDateTime(newPath, Date, Time);
            }
        }
        else
        {
            string newPath = folderPath + "\\" + FileInfo.name;
            bool ref = SetLocalFileTime(newPath.c_str(), "1995-01-01", "12:00:00");
            if (ref == true)
            {
                std::cout << "[*] 文件: " << newPath.c_str() << std::endl;
            }
        }

    } while (_findnext(Handle, &FileInfo) == 0);
    _findclose(Handle);
}

int main(int argc, char* argv[])
{
    BatchSetFileDateTime("D:\\lyshark", "1995-01-01", "0:0:0");

    system("pause");
    return 0;
}

运行后,目录下的文件将被重置时间戳;

本文参与?腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2023-11-20,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客?前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与?腾讯云自媒体分享计划? ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 11.8.1 切割文件名与路径
  • 11.8.2 遍历目录下文件
  • 11.8.3 重置文件目录时间戳
相关产品与服务
对象存储
对象存储(Cloud Object Storage,COS)是由腾讯云推出的无目录层次结构、无数据格式限制,可容纳海量数据且支持 HTTP/HTTPS 协议访问的分布式存储服务。腾讯云 COS 的存储桶空间无容量上限,无需分区管理,适用于 CDN 数据分发、数据万象处理或大数据计算与分析的数据湖等多种场景。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
http://www.vxiaotou.com