首先,我們需要對.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類,並傳入需要監控的目錄路徑,以及是否制定監控的文件類型(文章前面有所介紹)。
_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屬性訪問文件變更消息,同時也可以設置其是否需要將數據持久化到磁盤文件中。