C#實現對文件目錄的實時監控

  本文主要描述如何經過C#實現實時監控文件目錄下的變化,包括文件和目錄的添加,刪除,修改和重命名等操做。網絡

  首先,咱們須要對.net提供的FileSystemWatcher類有所瞭解。我有些懶,找了MSDN對該類的描述。app

  FileSystemWatcher類偵聽文件系統更改通知,並在目錄或目錄中的文件發生更改時引起事件。ide

  使用 FileSystemWatcher 監視指定目錄中的更改。可監視指定目錄中的文件或子目錄的更改。能夠建立一個組件來監視本地計算機、網絡驅動器或遠程計算機上的文件。ui

  若要監視全部文件中的更改,請將 Filter 屬性設置爲空字符串 ("") 或使用通配符(「*.*」)。若要監視特定的文件,請將 Filter 屬性設置爲該文件名。例如,若要監視文件 MyDoc.txt 中的更改,請將 Filter 屬性設置爲「MyDoc.txt」。也能夠監視特定類型文件中的更改。例如,若要監視文本文件中的更改,請將 Filter 屬性設置爲「*.txt」。spa

  可監視目錄或文件中的若干種更改。例如,可監視文件或目錄的 AttributesLastWrite 日期和時間或 Size 方面的更改。經過將 NotifyFilter 屬性設置爲 NotifyFilters 值之一來達到此目的。有關可監視的更改類型的更多信息,請參見 NotifyFilters操作系統

  可監視文件或目錄的重命名、刪除或建立。例如,若要監視文本文件的重命名,請將 Filter 屬性設置爲「*.txt」,並使用爲其參數指定的 Renamed 來調用 WaitForChanged 方法。.net

  Windows 操做系統在 FileSystemWatcher 建立的緩衝區中通知組件文件發生更改。若是短期內有不少更改,則緩衝區可能會溢出。這將致使組件失去對目錄更改的跟蹤,而且它將只提供通常性通知。使用 InternalBufferSize 屬性來增長緩衝區大小的開銷較大,由於它來自沒法換出到磁盤的非頁面內存,因此應確保緩衝區大小適中(儘可能小,但也要有足夠大小以便不會丟失任何文件更改事件)。若要避免緩衝區溢出,請使用 NotifyFilterIncludeSubdirectories 屬性,以即可以篩選掉不想要的更改通知。code

  使用 FileSystemWatcher 類時,請注意如下事項。component

  1) 對包括隱藏文件(夾)在內的全部文件(夾)進行監控。orm

  2) 您能夠爲 InternalBufferSize 屬性(用於監視網絡上的目錄)設置的最大大小爲 64 KB。

  FileSystemWatcher的實例監控到文件(夾)的變化後,會觸發相應的事件,其中文件(夾)的添加,刪除和修改會分別觸發Created,Deleted,Changed事件,文件(夾)重命名時觸發OnRenamed事件。

  

  而後,在熟悉了FileSystemWatcher類後,咱們開始本身的程序編寫。

  實例化FileSystemWatcher類,並傳入須要監控的目錄路徑,以及是否制定監控的文件類型(文章前面有所介紹)。

_watcher = new FileSystemWatcher(_path, _filter);

  註冊監聽事件,以及編寫事件觸發後相關的處理邏輯。

_watcher.Created += new FileSystemEventHandler(OnChanged);
            _watcher.Changed += new FileSystemEventHandler(OnChanged);
            _watcher.Deleted += new FileSystemEventHandler(OnChanged);
            _watcher.Renamed += new RenamedEventHandler(OnRenamed);
            _watcher.IncludeSubdirectories = true;
            _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 
 12         /// <summary>
 13         /// 監控是否正在運行
 14         /// </summary>
 15         public bool IsWatch
 16         {
 17             get
 18             {
 19                 return _isWatch;
 20             }
 21         }
 22 
 23         /// <summary>
 24         /// 文件變動信息隊列
 25         /// </summary>
 26         public CustomQueue<FileChangeInformation> FileChangeQueue
 27         {
 28             get
 29             {
 30                 return _queue;
 31             }
 32         }
 33 
 34         /// <summary>
 35         /// 初始化FileWatcher類
 36         /// </summary>
 37         /// <param name="path">監控路徑</param>
 38         public FileWatcher(string path)
 39         {
 40             _path = path;
 41             _queue = new CustomQueue<FileChangeInformation>();
 42         }
 43         /// <summary>
 44         /// 初始化FileWatcher類,並指定是否持久化文件變動消息
 45         /// </summary>
 46         /// <param name="path">監控路徑</param>
 47         /// <param name="isPersistence">是否持久化變動消息</param>
 48         /// <param name="persistenceFilePath">持久化保存路徑</param>
 49         public FileWatcher(string path, bool isPersistence, string persistenceFilePath)
 50         {
 51             _path = path;
 52             _queue = new CustomQueue<FileChangeInformation>(isPersistence, persistenceFilePath);
 53         }
 54 
 55         /// <summary>
 56         /// 初始化FileWatcher類,並指定是否監控指定類型文件
 57         /// </summary>
 58         /// <param name="path">監控路徑</param>
 59         /// <param name="filter">指定類型文件,格式如:*.txt,*.doc,*.rar</param>
 60         public FileWatcher(string path, string filter)
 61         {
 62             _path = path;
 63             _filter = filter;
 64             _queue = new CustomQueue<FileChangeInformation>();
 65         }
 66 
 67         /// <summary>
 68         /// 初始化FileWatcher類,並指定是否監控指定類型文件,是否持久化文件變動消息
 69         /// </summary>
 70         /// <param name="path">監控路徑</param>
 71         /// <param name="filter">指定類型文件,格式如:*.txt,*.doc,*.rar</param>
 72         /// <param name="isPersistence">是否持久化變動消息</param>
 73         /// <param name="persistenceFilePath">持久化保存路徑</param>
 74         public FileWatcher(string path, string filter, bool isPersistence, string persistenceFilePath)
 75         {
 76             _path = path;
 77             _filter = filter;
 78             _queue = new CustomQueue<FileChangeInformation>(isPersistence, persistenceFilePath);
 79         }
 80 
 81         /// <summary>
 82         /// 打開文件監聽器
 83         /// </summary>
 84         public void Open()
 85         {
 86             if (!Directory.Exists(_path))
 87             {
 88                 Directory.CreateDirectory(_path);
 89             }
 90 
 91             if (string.IsNullOrEmpty(_filter))
 92             {
 93                 _watcher = new FileSystemWatcher(_path);
 94             }
 95             else
 96             {
 97                 _watcher = new FileSystemWatcher(_path, _filter);
 98             }
 99             //註冊監聽事件
100             _watcher.Created += new FileSystemEventHandler(OnProcess);
101             _watcher.Changed += new FileSystemEventHandler(OnProcess);
102             _watcher.Deleted += new FileSystemEventHandler(OnProcess);
103             _watcher.Renamed += new RenamedEventHandler(OnFileRenamed);
104             _watcher.IncludeSubdirectories = true;
105             _watcher.EnableRaisingEvents = true;
106             _isWatch = true;
107         }
108 
109         /// <summary>
110         /// 關閉監聽器
111         /// </summary>
112         public void Close()
113         {
114             _isWatch = false;
115             _watcher.Created -= new FileSystemEventHandler(OnProcess);
116             _watcher.Changed -= new FileSystemEventHandler(OnProcess);
117             _watcher.Deleted -= new FileSystemEventHandler(OnProcess);
118             _watcher.Renamed -= new RenamedEventHandler(OnFileRenamed);
119             _watcher.EnableRaisingEvents = false;
120             _watcher = null;
121         }
122 
123         /// <summary>
124         /// 獲取一條文件變動消息
125         /// </summary>
126         /// <returns></returns>
127         public FileChangeInformation Get()
128         {
129             FileChangeInformation info = null;
130             if (_queue.Count > 0)
131             {
132                 lock (_queue)
133                 {
134                     info = _queue.Dequeue();
135                 }
136             }
137             return info;
138         }
139 
140         /// <summary>
141         /// 監聽事件觸發的方法
142         /// </summary>
143         /// <param name="sender"></param>
144         /// <param name="e"></param>
145         private void OnProcess(object sender, FileSystemEventArgs e)
146         {
147             try
148             {
149                 FileChangeType changeType = FileChangeType.Unknow;
150                 if (e.ChangeType == WatcherChangeTypes.Created)
151                 {
152                     if (File.GetAttributes(e.FullPath) == FileAttributes.Directory)
153                     {
154                         changeType = FileChangeType.NewFolder;
155                     }
156                     else
157                     {
158                         changeType = FileChangeType.NewFile;
159                     }
160                 }
161                 else if (e.ChangeType == WatcherChangeTypes.Changed)
162                 {
163                     //部分文件建立時一樣觸發文件變化事件,此時記錄變化操做沒有意義
164                     //若是
165                     if (_queue.SelectAll(
166                         delegate(FileChangeInformation fcm)
167                         {
168                             return fcm.NewPath == e.FullPath && fcm.ChangeType == FileChangeType.Change;
169                         }).Count<FileChangeInformation>() > 0)
170                     {
171                         return;
172                     }
173 
174                     //文件夾的變化,只針對建立,重命名和刪除動做,修改不作任何操做。
175                     //由於文件夾下任何變化一樣會觸發文件的修改操做,沒有任何意義.
176                     if (File.GetAttributes(e.FullPath) == FileAttributes.Directory)
177                     {
178                         return;
179                     }
180 
181                     changeType = FileChangeType.Change;
182                 }
183                 else if (e.ChangeType == WatcherChangeTypes.Deleted)
184                 {
185                     changeType = FileChangeType.Delete;
186                 }
187 
188                 //建立消息,並壓入隊列中
189                 FileChangeInformation info = new FileChangeInformation(Guid.NewGuid().ToString(), changeType, e.FullPath, e.FullPath, e.Name, e.Name);
190                 _queue.Enqueue(info);
191             }
192             catch
193             {
194                 Close();
195             }
196         }
197 
198         /// <summary>
199         /// 文件或目錄重命名時觸發的事件
200         /// </summary>
201         /// <param name="sender"></param>
202         /// <param name="e"></param>
203         private void OnFileRenamed(object sender, RenamedEventArgs e)
204         {
205             try
206             {
207                 //建立消息,並壓入隊列中
208                 FileChangeInformation info = new FileChangeInformation(Guid.NewGuid().ToString(), FileChangeType.Rename, e.OldFullPath, e.FullPath, e.OldName, e.Name);
209                 _queue.Enqueue(info);
210             }
211             catch
212             {
213                 Close();
214             }
215         }
216     }

  最後,功能調用以下:

 1 //初始化監控器
 2                 FileWatcher watcher = new FileWatcher(@"D:\");
 3                 watcher.Open();
 4 
 5                 FileChangeInformation fci = null;
 6                 //獲取消息
 7                 while (true)
 8                 {
 9                     //若是IsWatch爲False,則可能監控內部發生異常終止了監控,須要從新開啓監控
10                     if (watcher.IsWatch)
11                     {
12                         //隊列頂端的變動消息
13                         fci = watcher.Get();
14                         //處理消息的代碼
15                         //Print(fci);
16                     }
17                     else
18                     {
19                         watcher.Open();
20                     }
21                     Thread.Sleep(1000);
22                 }

  該程序實現了對文件目錄下全部子目錄和子文件的變化進行監控,並可經過FileChangeQueue屬性訪問文件變動消息,同時也能夠設置其是否須要將數據持久化到磁盤文件中。

  須要源碼的M我:GavinLeeChang@126.com

相關文章
相關標籤/搜索