如何编写代码监控特定路径下文件的变化?

2026-05-08 19:043阅读0评论SEO资讯
  • 内容介绍
  • 文章标签
  • 相关推荐

本文共计746个文字,预计阅读时间需要3分钟。

如何编写代码监控特定路径下文件的变化?

参考MSDN文档,以下是对函数`ReadDirectoryChangesW`和结构体`FILE_NOTIFY_INFORMATION`的简要介绍和代码示例:

函数简介:`ReadDirectoryChangesW`函数用于监视指定目录中的文件系统事件,如创建、删除、重命名等。该函数返回一个事件列表,供应用程序处理。

结构体简介:`FILE_NOTIFY_INFORMATION`结构体用于描述目录中的文件系统事件。它包含了事件类型、事件发生的时间戳、受影响文件的名称等信息。

代码示例:

cpp#include #include

int main() { HANDLE hDir=CreateFile(LC:\\path\\to\\directory, FILE_LIST_DIRECTORY, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL); if (hDir==INVALID_HANDLE_VALUE) { std::cerr << Failed to open directory. <

DWORD bytesReturned; FILE_NOTIFY_INFORMATION* pNotifyInfo=NULL; while (true) { if (!ReadDirectoryChangesW(hDir, &pNotifyInfo, sizeof(FILE_NOTIFY_INFORMATION), TRUE, FILE_NOTIFY_CHANGE_FILE_NAME | FILE_NOTIFY_CHANGE_DIR_NAME | FILE_NOTIFY_CHANGE_ATTRIBUTES | FILE_NOTIFY_CHANGE_SIZE | FILE_NOTIFY_CHANGE_LAST_WRITE | FILE_NOTIFY_CHANGE_SECURITY, &bytesReturned, NULL, NULL)) { std::cerr << Failed to read directory changes. <

std::wcout < pNotifyInfo=(FILE_NOTIFY_INFORMATION*)((char*)pNotifyInfo + pNotifyInfo->NextEntryOffset); }

CloseHandle(hDir); return 0;}

注意:- 在实际应用中,您可能需要根据需要修改事件类型和事件处理的逻辑。- 在循环中,需要不断读取事件,直到`ReadDirectoryChangesW`返回错误或应用程序需要停止监视。

参考MSDN文档

docs.microsoft.com/zh-cn/windows/desktop/api/winbase/nf-winbase-readdirectorychangesw

docs.microsoft.com/zh-cn/windows/desktop/api/winnt/ns-winnt-_file_notify_information

具体看代码

如何编写代码监控特定路径下文件的变化?

# include < iostream > # include < windows.h > # include < process.h > # include < tchar.h > # include < string > using namespace std; typedef void( * CHANGESCALLBACK)(const std::wstring strFileName); /* @ 监控指定目录下文件变化 @ strDirPath 需要监控的目录 @ dwChangesType 需要监控文件变化方式 @ onCallBack 变化后的回调处理函数 */ void CheckFilesChanges(const std::wstring strDirPath, DWORD dwChangesType, CHANGESCALLBACK onCallBack); void PrintMsg(const std::wstring strFileName) { wprintf_s(L "new add File:\t %s\r\n", strFileName.c_str()); } void CheckFilesChanges(const std::wstring strDirPath, DWORD dwChangesType, CHANGESCALLBACK onCallBack) { if (strDirPath.empty() || !onCallBack) return; HANDLE hDir = INVALID_HANDLE_VALUE; BYTE lpBuffer[1024]; ZeroMemory(lpBuffer, 1024); DWORD cbBytes = NULL; BOOL isOk = FALSE; FILE_NOTIFY_INFORMATION * pnotify = (FILE_NOTIFY_INFORMATION * )lpBuffer; FILE_NOTIFY_INFORMATION * tmp; ZeroMemory( & lpBuffer, sizeof(FILE_NOTIFY_INFORMATION)); hDir = CreateFile(strDirPath.c_str(), FILE_LIST_DIRECTORY, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OVERLAPPED, NULL); if (INVALID_HANDLE_VALUE == hDir) return; while (true) { isOk = ReadDirectoryChangesW(hDir, & lpBuffer, sizeof(lpBuffer), TRUE, FILE_NOTIFY_CHANGE_FILE_NAME | FILE_NOTIFY_CHANGE_LAST_WRITE, & cbBytes, NULL, NULL); if (isOk) { tmp = pnotify; PWSTR fileName = nullptr; std::wstring strFileName; if (tmp->FileNameLength) { DWORD dwLength = tmp->FileNameLength + 1; fileName = new wchar_t[dwLength]; if (fileName) { memset(fileName, 0, dwLength * 2); memcpy(fileName, tmp->FileName, dwLength * 2); strFileName = fileName; } if (fileName) delete []fileName; } if (tmp->Action == dwChangesType) { onCallBack(strFileName); } ZeroMemory(tmp, 1024); /*switch (tmp->Action){ case FILE_ACTION_ADDED: //当前目录新增文件{ onCallBack(strFileName); }break; case FILE_ACTION_REMOVED: //该文件已从目录中删除{ onCallBack(strFileName); } break; case FILE_ACTION_MODIFIED: //该文件已被修改。也可以是时间戳或属性的更改{ onCallBack(strFileName); } break; case FILE_ACTION_RENAMED_OLD_NAME: //该文件已重命名,这是旧名称。{ onCallBack(strFileName); } break; case FILE_ACTION_RENAMED_NEW_NAME: //该文件已重命名,这是新名称。{ onCallBack(strFileName); } break; default: break; }*/ } } CloseHandle(hDir); } unsigned int CALLBACK ThreadProc(void * arg) { CheckFilesChanges(_T("E:\\GoCode"), FILE_ACTION_ADDED, PrintMsg); return 0; } int main(int argc, char * argv[]) { HANDLE hThread = NULL; hThread = (HANDLE)::_beginthreadex(nullptr, NULL, ThreadProc, NULL, NULL, NULL); system("pause"); return 0; }

好了这篇文章就介绍到这了,需要的朋友可以参考一下。

本文共计746个文字,预计阅读时间需要3分钟。

如何编写代码监控特定路径下文件的变化?

参考MSDN文档,以下是对函数`ReadDirectoryChangesW`和结构体`FILE_NOTIFY_INFORMATION`的简要介绍和代码示例:

函数简介:`ReadDirectoryChangesW`函数用于监视指定目录中的文件系统事件,如创建、删除、重命名等。该函数返回一个事件列表,供应用程序处理。

结构体简介:`FILE_NOTIFY_INFORMATION`结构体用于描述目录中的文件系统事件。它包含了事件类型、事件发生的时间戳、受影响文件的名称等信息。

代码示例:

cpp#include #include

int main() { HANDLE hDir=CreateFile(LC:\\path\\to\\directory, FILE_LIST_DIRECTORY, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL); if (hDir==INVALID_HANDLE_VALUE) { std::cerr << Failed to open directory. <

DWORD bytesReturned; FILE_NOTIFY_INFORMATION* pNotifyInfo=NULL; while (true) { if (!ReadDirectoryChangesW(hDir, &pNotifyInfo, sizeof(FILE_NOTIFY_INFORMATION), TRUE, FILE_NOTIFY_CHANGE_FILE_NAME | FILE_NOTIFY_CHANGE_DIR_NAME | FILE_NOTIFY_CHANGE_ATTRIBUTES | FILE_NOTIFY_CHANGE_SIZE | FILE_NOTIFY_CHANGE_LAST_WRITE | FILE_NOTIFY_CHANGE_SECURITY, &bytesReturned, NULL, NULL)) { std::cerr << Failed to read directory changes. <

std::wcout < pNotifyInfo=(FILE_NOTIFY_INFORMATION*)((char*)pNotifyInfo + pNotifyInfo->NextEntryOffset); }

CloseHandle(hDir); return 0;}

注意:- 在实际应用中,您可能需要根据需要修改事件类型和事件处理的逻辑。- 在循环中,需要不断读取事件,直到`ReadDirectoryChangesW`返回错误或应用程序需要停止监视。

参考MSDN文档

docs.microsoft.com/zh-cn/windows/desktop/api/winbase/nf-winbase-readdirectorychangesw

docs.microsoft.com/zh-cn/windows/desktop/api/winnt/ns-winnt-_file_notify_information

具体看代码

如何编写代码监控特定路径下文件的变化?

# include < iostream > # include < windows.h > # include < process.h > # include < tchar.h > # include < string > using namespace std; typedef void( * CHANGESCALLBACK)(const std::wstring strFileName); /* @ 监控指定目录下文件变化 @ strDirPath 需要监控的目录 @ dwChangesType 需要监控文件变化方式 @ onCallBack 变化后的回调处理函数 */ void CheckFilesChanges(const std::wstring strDirPath, DWORD dwChangesType, CHANGESCALLBACK onCallBack); void PrintMsg(const std::wstring strFileName) { wprintf_s(L "new add File:\t %s\r\n", strFileName.c_str()); } void CheckFilesChanges(const std::wstring strDirPath, DWORD dwChangesType, CHANGESCALLBACK onCallBack) { if (strDirPath.empty() || !onCallBack) return; HANDLE hDir = INVALID_HANDLE_VALUE; BYTE lpBuffer[1024]; ZeroMemory(lpBuffer, 1024); DWORD cbBytes = NULL; BOOL isOk = FALSE; FILE_NOTIFY_INFORMATION * pnotify = (FILE_NOTIFY_INFORMATION * )lpBuffer; FILE_NOTIFY_INFORMATION * tmp; ZeroMemory( & lpBuffer, sizeof(FILE_NOTIFY_INFORMATION)); hDir = CreateFile(strDirPath.c_str(), FILE_LIST_DIRECTORY, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OVERLAPPED, NULL); if (INVALID_HANDLE_VALUE == hDir) return; while (true) { isOk = ReadDirectoryChangesW(hDir, & lpBuffer, sizeof(lpBuffer), TRUE, FILE_NOTIFY_CHANGE_FILE_NAME | FILE_NOTIFY_CHANGE_LAST_WRITE, & cbBytes, NULL, NULL); if (isOk) { tmp = pnotify; PWSTR fileName = nullptr; std::wstring strFileName; if (tmp->FileNameLength) { DWORD dwLength = tmp->FileNameLength + 1; fileName = new wchar_t[dwLength]; if (fileName) { memset(fileName, 0, dwLength * 2); memcpy(fileName, tmp->FileName, dwLength * 2); strFileName = fileName; } if (fileName) delete []fileName; } if (tmp->Action == dwChangesType) { onCallBack(strFileName); } ZeroMemory(tmp, 1024); /*switch (tmp->Action){ case FILE_ACTION_ADDED: //当前目录新增文件{ onCallBack(strFileName); }break; case FILE_ACTION_REMOVED: //该文件已从目录中删除{ onCallBack(strFileName); } break; case FILE_ACTION_MODIFIED: //该文件已被修改。也可以是时间戳或属性的更改{ onCallBack(strFileName); } break; case FILE_ACTION_RENAMED_OLD_NAME: //该文件已重命名,这是旧名称。{ onCallBack(strFileName); } break; case FILE_ACTION_RENAMED_NEW_NAME: //该文件已重命名,这是新名称。{ onCallBack(strFileName); } break; default: break; }*/ } } CloseHandle(hDir); } unsigned int CALLBACK ThreadProc(void * arg) { CheckFilesChanges(_T("E:\\GoCode"), FILE_ACTION_ADDED, PrintMsg); return 0; } int main(int argc, char * argv[]) { HANDLE hThread = NULL; hThread = (HANDLE)::_beginthreadex(nullptr, NULL, ThreadProc, NULL, NULL, NULL); system("pause"); return 0; }

好了这篇文章就介绍到这了,需要的朋友可以参考一下。