A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

转自:http://www.jb51.net/article/46299.htm
本文主要描述如何通过c#实现实时监控文件目录下的变化,包括文件和目录的添加,删除,修改和重命名等操作
首先,我们需要对.net提供的FileSystemWatcher类有所了解。我有些懒,找了MSDN对该类的描述。
FileSystemWatcher类侦听文件系统更改通知,并在目录或目录中的文件发生更改时引发事件。
使用 FileSystemWatcher 监视指定目录中的更改。可监视指定目录中的文件或子目录的更改。可以创建一个组件来监视本地计算机、网络驱动器或远程计算机上的文件。
若要监视所有文件中的更改,请将 Filter 属性设置为空字符串 ("") 或使用通配符(“*.*”)。若要监视特定的文件,请将 Filter 属性设置为该文件名。例如,若要监视文件 MyDoc.txt 中的更改,请将 Filter 属性设置为“MyDoc.txt”。也可以监视特定类型文件中的更改。例如,若要监视文本文件中的更改,请将 Filter 属性设置为“*.txt”。
可监视目录或文件中的若干种更改。例如,可监视文件或目录的 Attributes、LastWrite 日期和时间或 Size 方面的更改。通过将 NotifyFilter 属性设置为 NotifyFilters 值之一来达到此目的。有关可监视的更改类型的更多信息,请参见 NotifyFilters。
可监视文件或目录的重命名、删除或创建。例如,若要监视文本文件的重命名,请将 Filter 属性设置为“*.txt”,并使用为其参数指定的 Renamed 来调用 WaitForChanged 方法。
Windows 操作系统在 FileSystemWatcher 创建的缓冲区中通知组件文件发生更改。如果短时间内有很多更改,则缓冲区可能会溢出。这将导致组件失去对目录更改的跟踪,并且它将只提供一般性通知。使用 InternalBufferSize 属性来增加缓冲区大小的开销较大,因为它来自无法换出到磁盘的非页面内存,所以应确保缓冲区大小适中(尽量小,但也要有足够大小以便不会丢失任何文件更改事件)。若要避免缓冲区溢出,请使用 NotifyFilter 和 IncludeSubdirectories 属性,以便可以筛选掉不想要的更改通知。
使用 FileSystemWatcher 类时,请注意以下事项。
1) 对包括隐藏文件(夹)在内的所有文件(夹)进行监控。
2) 您可以为 InternalBufferSize 属性(用于监视网络上的目录)设置的最大大小为 64 KB。
FileSystemWatcher的实例监控到文件(夹)的变化后,会触发相应的事件,其中文件(夹)的添加,删除和修改会分别触发Created,Deleted,Changed事件,文件(夹)重命名时触发OnRenamed事件。
然后,在熟悉了FileSystemWatcher类后,我们开始自己的程序编写。
实例化FileSystemWatcher类,并传入需要监控的目录路径,以及是否制定监控的文件类型(文章前面有所介绍)。

  1. _watcher = new FileSystemWatcher(_path, _filter);
复制代码
注册监听事件,以及编写事件触发后相关的处理逻辑。

  1. _watcher.Created += new FileSystemEventHandler(OnChanged);
  2. _watcher.Changed += new FileSystemEventHandler(OnChanged);
  3. _watcher.Deleted += new FileSystemEventHandler(OnChanged);
  4. _watcher.Renamed += new RenamedEventHandler(OnRenamed);
  5. _watcher.IncludeSubdirectories = true;
  6. _watcher.EnableRaisingEvents = true;
复制代码

在本程序中,专门定义了一个FileChangeInformation类来记录文件变化信息,并定义了一个CustomQueue类,该类类似于Queue类,是一个数据先进先出的集合,用来存储所有的文件变化消息,并提供数据持久化功能。
监控类 - FileWatcher,代码如下:

  1. /// <summary>
  2. /// 文件监控类,用于监控指定目录下文件以及文件夹的变化
  3. /// </summary>
  4. public class FileWatcher
  5. {
  6. private FileSystemWatcher _watcher = null;
  7. private string _path = string.Empty;
  8. private string _filter = string.Empty;
  9. private bool _isWatch = false;
  10. private CustomQueue<FileChangeInformation> _queue = null;
  11. /// <summary>
  12. /// 监控是否正在运行
  13. /// </summary>
  14. public bool IsWatch
  15. {
  16. get
  17. {
  18. return _isWatch;
  19. }
  20. }
  21. /// <summary>
  22. /// 文件变更信息队列
  23. /// </summary>
  24. public CustomQueue<FileChangeInformation> FileChangeQueue
  25. {
  26. get
  27. {
  28. return _queue;
  29. }
  30. }
  31. /// <summary>
  32. /// 初始化FileWatcher类
  33. /// </summary>
  34. /// <param name="path">监控路径</param>
  35. public FileWatcher(string path)
  36. {
  37. _path = path;
  38. _queue = new CustomQueue<FileChangeInformation>();
  39. }
  40. /// <summary>
  41. /// 初始化FileWatcher类,并指定是否持久化文件变更消息
  42. /// </summary>
  43. /// <param name="path">监控路径</param>
  44. /// <param name="isPersistence">是否持久化变更消息</param>
  45. /// <param name="persistenceFilePath">持久化保存路径</param>
  46. public FileWatcher(string path, bool isPersistence, string persistenceFilePath)
  47. {
  48. _path = path;
  49. _queue = new CustomQueue<FileChangeInformation>(isPersistence, persistenceFilePath);
  50. }
  51. /// <summary>
  52. /// 初始化FileWatcher类,并指定是否监控指定类型文件
  53. /// </summary>
  54. /// <param name="path">监控路径</param>
  55. /// <param name="filter">指定类型文件,格式如:*.txt,*.doc,*.rar</param>
  56. public FileWatcher(string path, string filter)
  57. {
  58. _path = path;
  59. _filter = filter;
  60. _queue = new CustomQueue<FileChangeInformation>();
  61. }
  62. /// <summary>
  63. /// 初始化FileWatcher类,并指定是否监控指定类型文件,是否持久化文件变更消息
  64. /// </summary>
  65. /// <param name="path">监控路径</param>
  66. /// <param name="filter">指定类型文件,格式如:*.txt,*.doc,*.rar</param>
  67. /// <param name="isPersistence">是否持久化变更消息</param>
  68. /// <param name="persistenceFilePath">持久化保存路径</param>
  69. public FileWatcher(string path, string filter, bool isPersistence, string persistenceFilePath)
  70. {
  71. _path = path;
  72. _filter = filter;
  73. _queue = new CustomQueue<FileChangeInformation>(isPersistence, persistenceFilePath);
  74. }
  75. /// <summary>
  76. /// 打开文件监听器
  77. /// </summary>
  78. public void Open()
  79. {
  80. if (!Directory.Exists(_path))
  81. {
  82. Directory.CreateDirectory(_path);
  83. }
  84. if (string.IsNullOrEmpty(_filter))
  85. {
  86. _watcher = new FileSystemWatcher(_path);
  87. }
  88. else
  89. {
  90. _watcher = new FileSystemWatcher(_path, _filter);
  91. }
  92. //注册监听事件
  93. _watcher.Created += new FileSystemEventHandler(OnProcess);
  94. _watcher.Changed += new FileSystemEventHandler(OnProcess);
  95. _watcher.Deleted += new FileSystemEventHandler(OnProcess);
  96. _watcher.Renamed += new RenamedEventHandler(OnFileRenamed);
  97. _watcher.IncludeSubdirectories = true;
  98. _watcher.EnableRaisingEvents = true;
  99. _isWatch = true;
  100. }
  101. /// <summary>
  102. /// 关闭监听器
  103. /// </summary>
  104. public void Close()
  105. {
  106. _isWatch = false;
  107. _watcher.Created -= new FileSystemEventHandler(OnProcess);
  108. _watcher.Changed -= new FileSystemEventHandler(OnProcess);
  109. _watcher.Deleted -= new FileSystemEventHandler(OnProcess);
  110. _watcher.Renamed -= new RenamedEventHandler(OnFileRenamed);
  111. _watcher.EnableRaisingEvents = false;
  112. _watcher = null;
  113. }
  114. /// <summary>
  115. /// 获取一条文件变更消息
  116. /// </summary>
  117. /// <returns></returns>
  118. public FileChangeInformation Get()
  119. {
  120. FileChangeInformation info = null;
  121. if (_queue.Count > 0)
  122. {
  123. lock (_queue)
  124. {
  125. info = _queue.Dequeue();
  126. }
  127. }
  128. return info;
  129. }
  130. /// <summary>
  131. /// 监听事件触发的方法
  132. /// </summary>
  133. /// <param name="sender"></param>
  134. /// <param name="e"></param>
  135. private void OnProcess(object sender, FileSystemEventArgs e)
  136. {
  137. try
  138. {
  139. FileChangeType changeType = FileChangeType.Unknow;
  140. if (e.ChangeType == WatcherChangeTypes.Created)
  141. {
  142. if (File.GetAttributes(e.FullPath) == FileAttributes.Directory)
  143. {
  144. changeType = FileChangeType.NewFolder;
  145. }
  146. else
  147. {
  148. changeType = FileChangeType.NewFile;
  149. }
  150. }
  151. else if (e.ChangeType == WatcherChangeTypes.Changed)
  152. {
  153. //部分文件创建时同样触发文件变化事件,此时记录变化操作没有意义
  154. //如果
  155. if (_queue.SelectAll(
  156. delegate(FileChangeInformation fcm)
  157. {
  158. return fcm.NewPath == e.FullPath && fcm.ChangeType == FileChangeType.Change;
  159. }).Count<FileChangeInformation>() > 0)
  160. {
  161. return;
  162. }
  163. //文件夹的变化,只针对创建,重命名和删除动作,修改不做任何操作。
  164. //因为文件夹下任何变化同样会触发文件的修改操作,没有任何意义.
  165. if (File.GetAttributes(e.FullPath) == FileAttributes.Directory)
  166. {
  167. return;
  168. }
  169. changeType = FileChangeType.Change;
  170. }
  171. else if (e.ChangeType == WatcherChangeTypes.Deleted)
  172. {
  173. changeType = FileChangeType.Delete;
  174. }
  175. //创建消息,并压入队列中
  176. FileChangeInformation info = new FileChangeInformation(Guid.NewGuid().ToString(), changeType, e.FullPath, e.FullPath, e.Name, e.Name);
  177. _queue.Enqueue(info);
  178. }
  179. catch
  180. {
  181. Close();
  182. }
  183. }
  184. /// <summary>
  185. /// 文件或目录重命名时触发的事件
  186. /// </summary>
  187. /// <param name="sender"></param>
  188. /// <param name="e"></param>
  189. private void OnFileRenamed(object sender, RenamedEventArgs e)
  190. {
  191. try
  192. {
  193. //创建消息,并压入队列中
  194. FileChangeInformation info = new FileChangeInformation(Guid.NewGuid().ToString(), FileChangeType.Rename, e.OldFullPath, e.FullPath, e.OldName, e.Name);
  195. _queue.Enqueue(info);
  196. }
  197. catch
  198. {
  199. Close();
  200. }
  201. }
  202. }
复制代码

最后,功能调用如下:

  1. //初始化监控器
  2. FileWatcher watcher = new FileWatcher(@"D:\");
  3. watcher.Open();
  4. FileChangeInformation fci = null;
  5. //获取消息
  6. while (true)
  7. {
  8. //如果IsWatch为False,则可能监控内部发生异常终止了监控,需要重新开启监控
  9. if (watcher.IsWatch)
  10. {
  11. //队列顶端的变更消息
  12. fci = watcher.Get();
  13. //处理消息的代码
  14. //Print(fci);
  15. }
  16. else
  17. {
  18. watcher.Open();
  19. }
  20. Thread.Sleep(1000);
  21. }
复制代码



1 个回复

倒序浏览
该程序实现了对文件目录下所有子目录和子文件的变化进行监控,并可通过FileChangeQueue属性访问文件变更消息,同时也可以设置其是否需要将数据持久化到磁盘文件中。
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马