1. // .h文件
  2. #pragma once
  3. typedef void (*PFN_NotifyAction)(DWORD dwAction, LPWSTR szFile, DWORD dwLength);
  4. class CDirectoryWatch
  5. {
  6. public:
  7. CDirectoryWatch(void);
  8. virtual ~CDirectoryWatch(void);
  9. public:
  10. BOOL StartDirectoryWatch(LPCTSTR lpszDirectory, PFN_NotifyAction pFn_NotifyAction);
  11. BOOL StopDirectoryWatch(void);
  12. private:
  13. static UINT __cdecl ThreadProc(LPVOID lParam);
  14. static UINT __cdecl DirectoryWatch(LPVOID lParam);
  15. private:
  16. HANDLE m_hFile;
  17. CWinThread* m_pThread;
  18. TCHAR m_szDirectory[MAX_PATH];
  19. };
  1. // .cpp文件
  2. #include "StdAfx.h"
  3. #include "DirectoryWatch.h"
  4. #include <strsafe.h>
  5. typedef enum
  6. {
  7. MSG_STARTWATCH = (WM_USER + 0x11),
  8. MSG_STOPWATCH,
  9. MSG_EXITTHREAD
  10. };
  11. #define MAX_BUFFER_SIZE (1024)
  12. typedef struct _tagWATCHPARAMETERS
  13. {
  14. _tagWATCHPARAMETERS()
  15. {
  16. hFile = INVALID_HANDLE_VALUE;
  17. hEvent = NULL;
  18. memset(&ol, 0, sizeof(OVERLAPPED));
  19. pBuffer = NULL;
  20. dwBufferSize = 0;
  21. bExit = FALSE;
  22. pFn_NotifyAction = NULL;
  23. }
  24. HANDLE hFile;
  25. HANDLE hEvent;
  26. OVERLAPPED ol;
  27. BYTE* pBuffer;
  28. DWORD dwBufferSize;
  29. BOOL bExit;
  30. PFN_NotifyAction pFn_NotifyAction;
  31. }WATCH_PARAMETERS, *PWATCH_PARAMETERS;
  32. CDirectoryWatch::CDirectoryWatch() : m_hFile(INVALID_HANDLE_VALUE), m_pThread(NULL)
  33. {
  34. memset(m_szDirectory, 0, sizeof(m_szDirectory));
  35. m_pThread = AfxBeginThread(ThreadProc, NULL, 0, CREATE_SUSPENDED, 0, NULL);
  36. if(NULL == m_pThread)
  37. {
  38. TRACE("Error Code : %d\n", GetLastError());
  39. return ;
  40. }
  41. m_pThread->m_bAutoDelete = FALSE;
  42. m_pThread->ResumeThread();
  43. }
  44. CDirectoryWatch::~CDirectoryWatch()
  45. {
  46. if(INVALID_HANDLE_VALUE != m_hFile)
  47. {
  48. CloseHandle(m_hFile);
  49. m_hFile = INVALID_HANDLE_VALUE;
  50. }
  51. if((NULL != m_pThread) && (NULL != m_pThread->m_hThread))
  52. {
  53. m_pThread->PostThreadMessage(MSG_EXITTHREAD, 0, 0);
  54. WaitForSingleObject(m_pThread->m_hThread, INFINITE);
  55. delete m_pThread;
  56. m_pThread = NULL;
  57. }
  58. }
  59. BOOL CDirectoryWatch::StartDirectoryWatch(LPCTSTR lpszDirectory, PFN_NotifyAction pFn_NotifyAction)
  60. {
  61. if(NULL == m_pThread)
  62. {
  63. return FALSE;
  64. }
  65. if(NULL == lpszDirectory)
  66. {
  67. return FALSE;
  68. }
  69. if(NULL == pFn_NotifyAction)
  70. {
  71. return FALSE;
  72. }
  73. if(!PathFileExists(lpszDirectory))
  74. {
  75. TRACE("Error Code : %d\n", GetLastError());
  76. return FALSE;
  77. }
  78. if(!PathIsDirectory(lpszDirectory))
  79. {
  80. TRACE("Error Code : %d\n", GetLastError());
  81. return FALSE;
  82. }
  83. if(0 == _tcslen(m_szDirectory))
  84. {
  85. StringCchPrintf(m_szDirectory, _countof(m_szDirectory), _T("%s"), lpszDirectory);
  86. }
  87. else if(CSTR_EQUAL != CompareStringOrdinal(m_szDirectory, -1, lpszDirectory, -1, TRUE))
  88. {
  89. TRACE("Not Change Directory.\n");
  90. return FALSE;
  91. }
  92. if(INVALID_HANDLE_VALUE == m_hFile)
  93. {
  94. m_hFile = CreateFile(lpszDirectory, FILE_LIST_DIRECTORY, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
  95. NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OVERLAPPED, NULL);
  96. if(INVALID_HANDLE_VALUE == m_hFile)
  97. {
  98. TRACE("Error Code : %d\n", GetLastError());
  99. return FALSE;
  100. }
  101. }
  102. return m_pThread->PostThreadMessage(MSG_STARTWATCH, (WPARAM)m_hFile, (LPARAM)pFn_NotifyAction);
  103. }
  104. BOOL CDirectoryWatch::StopDirectoryWatch()
  105. {
  106. if(NULL != m_pThread)
  107. {
  108. return m_pThread->PostThreadMessage(MSG_STOPWATCH, 0, 0);
  109. }
  110. return FALSE;
  111. }
  112. UINT __cdecl CDirectoryWatch::DirectoryWatch(LPVOID lParam)
  113. {
  114. WATCH_PARAMETERS* pParam = (WATCH_PARAMETERS*)lParam;
  115. if(NULL == pParam)
  116. {
  117. return 0;
  118. }
  119. HANDLE& hFile = pParam->hFile;
  120. BYTE* pBuffer = pParam->pBuffer;
  121. DWORD dwBufferSize = pParam->dwBufferSize;
  122. OVERLAPPED& ol = pParam->ol;
  123. HANDLE& hEvent = pParam->hEvent;
  124. BOOL& bExit = pParam->bExit;
  125. PFN_NotifyAction pFn_NotifyAction = pParam->pFn_NotifyAction;
  126. DWORD dwBytesReturn = 0;
  127. DWORD dwRet = WAIT_FAILED;
  128. DWORD dwOffSet = 0;
  129. TCHAR szFile[MAX_PATH] = {0};
  130. while(TRUE)
  131. {
  132. if(WAIT_OBJECT_0 != WaitForSingleObject(hEvent, INFINITE))
  133. {
  134. TRACE("Error Code : %d\n", GetLastError());
  135. break;
  136. }
  137. if(bExit)
  138. {
  139. break;
  140. }
  141. if(!ReadDirectoryChangesW(hFile, pBuffer, dwBufferSize, TRUE,
  142. FILE_NOTIFY_CHANGE_FILE_NAME | FILE_NOTIFY_CHANGE_DIR_NAME | FILE_NOTIFY_CHANGE_ATTRIBUTES
  143. | FILE_NOTIFY_CHANGE_SIZE | FILE_NOTIFY_CHANGE_LAST_WRITE | FILE_NOTIFY_CHANGE_LAST_ACCESS
  144. | FILE_NOTIFY_CHANGE_CREATION | FILE_NOTIFY_CHANGE_SECURITY, &dwBytesReturn, &ol, NULL))
  145. {
  146. TRACE("Error Code : %d\n", GetLastError());
  147. break;
  148. }
  149. if(!GetOverlappedResult(hFile, &ol, &dwBytesReturn, TRUE))
  150. {
  151. TRACE("Error Code : %d\n", GetLastError());
  152. break;
  153. }
  154. FILE_NOTIFY_INFORMATION* pFileNotify = (FILE_NOTIFY_INFORMATION*)pBuffer;
  155. do
  156. {
  157. if(pFn_NotifyAction && (WAIT_OBJECT_0 == WaitForSingleObject(hEvent, 0)))
  158. {
  159. pFn_NotifyAction(pFileNotify->Action, pFileNotify->FileName, (pFileNotify->FileNameLength) / sizeof(WCHAR));
  160. }
  161. dwOffSet = pFileNotify->NextEntryOffset;
  162. pFileNotify = (FILE_NOTIFY_INFORMATION*)((BYTE*)pFileNotify + dwOffSet);
  163. } while (dwOffSet);
  164. }
  165. TRACE0("DirectoryWatch Thread Exit ... \n");
  166. return 0;
  167. }
  168. UINT __cdecl CDirectoryWatch::ThreadProc(LPVOID lParam)
  169. {
  170. WATCH_PARAMETERS* pParam = new WATCH_PARAMETERS;
  171. if(NULL == pParam)
  172. {
  173. goto __CLEANUP__;
  174. }
  175. BYTE* pBuffer = new BYTE[MAX_BUFFER_SIZE];
  176. if(NULL == pBuffer)
  177. {
  178. goto __CLEANUP__;
  179. }
  180. memset(pBuffer, 0, MAX_BUFFER_SIZE);
  181. pParam->pBuffer = pBuffer;
  182. pParam->dwBufferSize = MAX_BUFFER_SIZE;
  183. HANDLE hWatchEvent  = CreateEvent(NULL, TRUE, FALSE, NULL);
  184. if(NULL == hWatchEvent)
  185. {
  186. goto __CLEANUP__;
  187. }
  188. pParam->ol.hEvent = hWatchEvent;
  189. CWinThread* pThread = NULL;
  190. HANDLE hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
  191. if(NULL == hEvent)
  192. {
  193. goto __CLEANUP__;
  194. }
  195. pParam->hEvent = hEvent;
  196. MSG msg;
  197. while(GetMessage(&msg, NULL, 0, 0))
  198. {
  199. switch(msg.message)
  200. {
  201. case MSG_STARTWATCH:
  202. {
  203. HANDLE hFile = (HANDLE)(msg.wParam);
  204. PFN_NotifyAction pFn_NotifyAction = (PFN_NotifyAction)(msg.lParam);
  205. if((INVALID_HANDLE_VALUE == hFile) && (NULL == pFn_NotifyAction))
  206. {
  207. break;
  208. }
  209. if(NULL == pThread)
  210. {
  211. pParam->hFile = hFile;
  212. pParam->pFn_NotifyAction = pFn_NotifyAction;
  213. pThread = AfxBeginThread(DirectoryWatch, (LPVOID)pParam, 0, CREATE_SUSPENDED, NULL);
  214. if(NULL == pThread)
  215. {
  216. goto __CLEANUP__;
  217. }
  218. pThread->m_bAutoDelete = FALSE;
  219. pThread->ResumeThread();
  220. }
  221. SetEvent(hEvent);
  222. }
  223. break;
  224. case MSG_STOPWATCH:
  225. {
  226. ResetEvent(hEvent);
  227. }
  228. break;
  229. case MSG_EXITTHREAD:
  230. {
  231. SetEvent(hEvent);
  232. pParam->bExit = FALSE;
  233. if((NULL != pThread) && (NULL != pThread->m_hThread))
  234. {
  235. WaitForSingleObject(pThread->m_hThread, INFINITE);
  236. delete pThread;
  237. pThread = NULL;
  238. }
  239. goto __CLEANUP__;
  240. }
  241. default:
  242. break;
  243. }
  244. TranslateMessage(&msg);
  245. DispatchMessage(&msg);
  246. }
  247. __CLEANUP__:
  248. if(NULL != hWatchEvent)
  249. {
  250. CloseHandle(hWatchEvent);
  251. hWatchEvent = NULL;
  252. }
  253. if(NULL != pBuffer)
  254. {
  255. delete[] pBuffer;
  256. pBuffer = NULL;
  257. }
  258. if(NULL != pParam)
  259. {
  260. delete pParam;
  261. pParam = NULL;
  262. }
  263. TRACE0("ThreadProc Thread Exit ...\n");
  264. return 0;
  265. }
  1. // 测试代码
  2. #include "stdafx.h"
  3. #include "DirectoryWatch.h"
  4. void NotifyAction(DWORD dwAction, LPWSTR szFile, DWORD dwLength)
  5. {
  6. switch(dwAction)
  7. {
  8. case FILE_ACTION_ADDED:
  9. wprintf(L"FILE_ACTION_ADDED: \n\t");
  10. break;
  11. case FILE_ACTION_REMOVED:
  12. wprintf(L"FILE_ACTION_REMOVED: \n\t");
  13. break;
  14. case FILE_ACTION_MODIFIED:
  15. wprintf(L"FILE_ACTION_MODIFIED: \n\t");
  16. break;
  17. case FILE_ACTION_RENAMED_OLD_NAME:
  18. wprintf(L"FILE_ACTION_RENAMED_OLD_NAME: \n\t");
  19. break;
  20. case FILE_ACTION_RENAMED_NEW_NAME:
  21. wprintf(L"FILE_ACTION_RENAMED_NEW_NAME: \n\t");
  22. break;
  23. default:
  24. break;
  25. }
  26. WCHAR szPath[MAX_PATH] = {0};
  27. wmemcpy(szPath, szFile, min(dwLength, MAX_PATH));
  28. wprintf(L"%s\n", szPath);
  29. }
  30. int _tmain(int argc, TCHAR* argv[], TCHAR* envp[])
  31. {
  32. CDirectoryWatch watch;
  33. wprintf(L"Start Directory Watch ...\n");
  34. watch.StartDirectoryWatch(_T("F:\\11"), NotifyAction);
  35. Sleep(30 * 1000);
  36. watch.StopDirectoryWatch();
  37. wprintf(L"Stop Directory Watch ...\n");
  38. Sleep(10 * 1000);
  39. wprintf(L"Start Directory Watch ...\n");
  40. watch.StartDirectoryWatch(_T("F:\\11"), NotifyAction);
  41. Sleep(30 * 1000);
  42. watch.StopDirectoryWatch();
  43. wprintf(L"Stop Directory Watch ...\n");
  44. Sleep(30 * 1000);
  45. wprintf(L"Process Exit ...\n");
  46. return 0;
  47. }

效果如下图所示:

http://blog.csdn.net/visualeleven/article/details/7562014

ReadDirectoryChangesW 监控文件夹 (一个简单的监控示例程序)(文件被修改了,也可以探测到)的更多相关文章

  1. gulp插件实现压缩一个文件夹下不同目录下的js文件(支持es6)

    gulp-uglify:压缩js大小,只支持es5 安装: cnpm: cnpm i gulp-uglify -D yarn: yarn add gulp-uglify -D 使用: 代码实现1:压缩 ...

  2. META-INF文件夹是干啥的,META-INF文件夹的作用, META-INF文件夹能删吗

    今天有人问到 META-INF文件夹是干啥的,META-INF文件夹的作用, META-INF文件夹能删吗,还有项目的META-INF下面一般会有个MANIFEST.MF 文件,都是干啥的. 百度搜了 ...

  3. selector是在文件夹drawable中进行定义的xml文件。

    获取Drawable对象: Resources res = mContext.getResources(); Drawable myImage = res.getDrawable(R.drawable ...

  4. 导出不带.svn的文件夹或者是不含.class的文件

    转载自:http://blog.csdn.net/z278718149/article/details/21537395 如何导出不带.svn的文件夹或者是不含.class的文件 在工作环境中,有的时 ...

  5. selector是在文件夹drawable中进行定义的xml文件转载 https://www.cnblogs.com/fx2008/p/3157040.html

    获取Drawable对象: Resources res = mContext.getResources(); Drawable myImage = res.getDrawable(R.drawable ...

  6. 一个简单的P2P传输程序

    写了一个简单的P2P传输程序,在P2P的圈子中传输文件,不过为了简便,这个程序没有真正的传输文件,只是简单的判断一下文件的位置在哪里.这个程序可以处理当有一个peer闪退的情况,在这种情况下,剩下的p ...

  7. Android加载SD卡目录,文件夹遍历,图片设置,设置文件对应打开方式等

    此案例主要说的是Android使用GridView加载SD卡下所有目录,文件夹多层遍历,文件图标修改,设置文件对应打开方式等功能. 如图: 代码: public class GridViewFile ...

  8. IOS开发之小实例--使用UIImagePickerController创建一个简单的相机应用程序

    前言:本篇博文是本人阅读国外的IOS Programming Tutorial的一篇入门文章的学习过程总结,难度不大,因为是入门.主要是入门UIImagePickerController这个控制器,那 ...

  9. iOS开发UI篇—使用嵌套模型完成的一个简单汽车图标展示程序

    iOS开发UI篇—使用嵌套模型完成的一个简单汽车图标展示程序 一.plist文件和项目结构图 说明:这是一个嵌套模型的示例 二.代码示例: YYcarsgroup.h文件代码: // // YYcar ...

随机推荐

  1. linux下配置Apache基于加密的认证访问

    1.首先要确认安装了 mod_ssl模块 我的机器是centos是系统,执行下面命令 yum install -y mod_ssl 2.用openssl工具生成密钥,证书请求文件,证书 在/usr/l ...

  2. SQL中char,varchar,nvarchar等的异同

    比较这几个数据类型,总是忘记,可能比较细节的原因.先做个记号,回头完善.

  3. Python原始套接字编程

    在实验中需要自己构造单独的HTTP数据报文,而使用SOCK_STREAM进行发送数据包,需要进行完整的TCP交互. 因此想使用原始套接字进行编程,直接构造数据包,并在IP层进行发送,即采用SOCK_R ...

  4. 【转】你需要知道的Python用法

    在使用Python多年以后,我偶然发现了一些我们过去不知道的功能和特性.一些可以说是非常有用,但却没有充分利用.考虑到这一点,我编辑了一些的你应该了解的Pyghon功能特色. 带任意数量参数的函数 你 ...

  5. #ifdef预编译相关用法

    #ifdef预编译相关用法主要有:(1)#ifdef XXX executing the corresponding xxx code #endif(2)#ifdef XXX executing th ...

  6. C# 反射学习总结

    C#中的反射可以使得程序集和类型(类.结构.委托.接口和枚举)以及类型中的成员(方法.字段.属性.事件.参数.构造函数等)都成为变量在编程中动态调用.

  7. 通过keepalived实现 MySQL VIP 自动切换

    首先配置keepalived.链接如下:http://blog.itpub.net/28939273/viewspace-1808369/ 主服务器keepalived的配置文件内容如下: [root ...

  8. 让别人也可以访问你电脑上的ASP.NET MVC创建的网站

    最近在写一个网站,昨天刚写完,由于要和朋友一起测试,但是他电脑上没有环境,所以希望我在自己电脑上部署一下,让他直接通过浏览器来访问来测试,所以从昨晚到今天上午,通过各种搜索,终于搞定了. 先介绍一下我 ...

  9. Google protobuf安装

    1:需要安装sudo apt-get install x11-apps libwayland-ltst-client0 libtxc-dxtn-s2tc0 x11-session-utils  x11 ...

  10. 从零开始学ios开发(六):IOS控件(3),Segmented Control、Switch

    这次的学习还是基于上一个项目继续进行(你也可以新建一个项目)学习Segmented Control和Switch. Segmented Control Switch Segmented Control ...