谈谈C#文件监控对象FileSystemWatcher使用感受

http://www.cnblogs.com/zhaojingjing/archive/2011/01/21/1941586.html

 

最近在项目中有这么个需求,就是得去实时获取某个在无规律改变的文本文件中的内容。首先想到的是用程序定期去访问这个文件,因为对实时性要求很高,间隔不能超过1S,而且每次获取到文本内容都要去分发给WEB服务器做别的操作,而那个文本的写入有时候会频繁,1秒可能多次,但是也有可能在相当长一段时间内是没有任何写入的。

这样一来如果每秒都去访问文件的话,一个是IO问题,还有就是每次操作都会引起后端一系列程序的反应,文本在长时间内无写入的话,一秒一次的触发一系列徒劳的事情太不可取了。

最终发现了c#中的FileSystemWatcher对象,在应用FileSystemWatcher之前,首先了解一下这个对象的基本属性和事件,首先普及一下FileSystemWatcher基本知识。

FileSystemWatcher基础

属性:

    Path——这个属性告诉FileSystemWatcher它需要监控哪条路径。例如,如果我们将这个属性设为“C:/test”,对象就监控test目录下所有文件发生的所有改变(包括删除,修改,创建,重命名)。

    IncludeSubDirectories——这个属性说明FileSystemWatcher对象是否应该监控子目录中(所有文件)发生的改变。

    Filter——这个属性允许你过滤掉某些类型的文件发生的变化。例如,如果我们只希望在TXT文件被修改/新建/删除时提交通知,可以将这个属性设为“*txt”。在处理高流量或大型目录时,使用这个属性非常方便。

NotifyFilter——获取或设置要监视的更改类型。可以进一步的过滤要监控的更改类型,如watcher.NotifyFilter = NotifyFilters.LastAccess | NotifyFilters.LastWrite

           | NotifyFilters.FileName | NotifyFilters.DirectoryName;

事件:

    Changed——当被监控的目录中有一个文件被修改时,就提交这个事件。值得注意的是,这个事件可能会被提交多次,即使文件的内容仅仅发生一项改变。这是由于在保存文件时,文件的其它属性也发生了改变。

    Created——当被监控的目录新建一个文件时,就提交这个事件。如果你计划用这个事件移动新建的事件,你必须在事件处理器中写入一些错误处理代码,它能处理当前文件被其它进程使用的情况。之所以要这样做,是因为Created事件可能在建立文件的进程释放文件之前就被提交。如果你没有准备正确处理这种情况的代码,就可能出现异常。

    Deleted——当被监控的目录中有一个文件被删除,就提交这个事件。

    Renamed——当被监控的目录中有一个文件被重命名,就提交这个事件。 

 

注:如果你没有将EnableRaisingEvents设为真,系统不会提交任何一个事件。如果有时FileSystemWatcher对象似乎无法工作,请首先检查EnableRaisingEvents,确保它被设为真。

 

事件处理

 

FileSystemWatcher调用一个事件处理器时,它包含两个自变量——一个叫做“sender”的对象和一个叫做“e”的 FileSystemEventArgs对象。我们感兴趣的自变量为FileSystemEventArgs自变量。这个对象中包含有提交事件的原因。以下是FileSystemEventArgs对象的一些属性:

 

属性:

 

  Name——这个属性中使事件被提交的文件的名称。其中并不包含文件的路径——只包含使用事件被提交的文件或目录名称。

  ChangeType——这是一个WatcherChangeTypes,它指出要提交哪个类型的事件。其有效值包括:

  Changed

  Created

  Deleted

  Renamed

  FullPath——这个属性中包含使事件被提交的文件的完整路径,包括文件名和目录名。

 

注意:FileSystemEventArgs对象是监控文件夹下有文件创建、删除、修改时的自变量,如果是重命名的话为RenamedEventArgs对象此时除了FileSystemEventArgs对象的属性值,多了一个OldFullPath,为重命名之前的文件名。

 

以上为FileSystemEventArgs的基本知识,大部分是从网上搜找的然后自己稍微整理了一下。

 

下面为简单用法:

 

 

 

 

01using System;
02using System.IO;
03  
04namespace test
05{
06    class Program
07    {
08        static void Main(string[] args)
09        {
10  
11  
12            WatcherStrat(@"C:/test", "*.txt");
13            //由于是控制台程序,加个输入避免主线程执行完毕,看不到监控效果
14            Console.ReadKey();
15  
16        }
17  
18        
19  
20        private static void WatcherStrat(string path, string filter)
21        {
22            FileSystemWatcher watcher = new FileSystemWatcher();
23            watcher.Path = path;
24  
25            watcher.Filter = filter;
26  
27            watcher.Changed += new FileSystemEventHandler(OnProcess);
28            watcher.Created += new FileSystemEventHandler(OnProcess);
29            watcher.Deleted += new FileSystemEventHandler(OnProcess);
30            watcher.Renamed += new RenamedEventHandler(OnRenamed);
31  
32            watcher.EnableRaisingEvents = true;
33        }
34  
35  
36  
37  
38        private static void OnProcess(object source, FileSystemEventArgs e)
39        
40            if (e.ChangeType == WatcherChangeTypes.Created)
41            {
42                OnCreated(source, e);
43  
44            }
45            else if (e.ChangeType == WatcherChangeTypes.Changed)
46            {
47                OnChanged(source, e);
48  
49            }
50            else if (e.ChangeType == WatcherChangeTypes.Deleted)
51            {
52                OnDeleted(source, e);
53  
54            }
55        }
56  
57        private static void OnCreated(object source, FileSystemEventArgs e)
58        {
59  
60            Console.WriteLine("文件新建事件处理逻辑");
61              
62        }
63  
64        private static void OnChanged(object source, FileSystemEventArgs e)
65        {
66  
67            Console.WriteLine("文件改变事件处理逻辑");
68        }
69  
70        private static void OnDeleted(object source, FileSystemEventArgs e)
71        {
72  
73            Console.WriteLine("文件删除事件处理逻辑");
74        }
75  
76        private static void OnRenamed(object source, RenamedEventArgs e)
77        {
78  
79            Console.WriteLine("文件重命名事件处理逻辑");
80        }
81  
82    }
83}

 

 

 

 

 

 

用上面的方法会发现,在一次文本文件变化的时候OnChanged事件会触发两次,这是因为除了文本内容变化之外还有文件其他的属性也变化了例如修改时间。

     

     为了解决这问题,也便于项目当中实际使用,写了下面几个类来实际使用:

 

 主方法:

 

 

01using System;
02using System.IO;
03  
04namespace test
05{
06    class Program
07    {
08        static void Main(string[] args)
09        {
10  
11  
12  
13            MyFileSystemWather myWather = new MyFileSystemWather(@"C:/test", "*.txt");
14            myWather.OnChanged += new FileSystemEventHandler(OnChanged);
15            myWather.OnCreated += new FileSystemEventHandler(OnCreated);
16            myWather.OnRenamed += new RenamedEventHandler(OnRenamed);
17            myWather.OnDeleted += new FileSystemEventHandler(OnDeleted);
18            myWather.Start();
19            //由于是控制台程序,加个输入避免主线程执行完毕,看不到监控效果
20            Console.ReadKey();
21  
22        }
23  
24        private static void OnCreated(object source, FileSystemEventArgs e)
25        {
26  
27            Console.WriteLine("文件新建事件处理逻辑");
28              
29        }
30  
31        private static void OnChanged(object source, FileSystemEventArgs e)
32        {
33  
34            Console.WriteLine("文件改变事件处理逻辑");
35        }
36  
37        private static void OnDeleted(object source, FileSystemEventArgs e)
38        {
39  
40            Console.WriteLine("文件删除事件处理逻辑");
41        }
42  
43        private static void OnRenamed(object source, RenamedEventArgs e)
44        {
45  
46            Console.WriteLine("文件重命名事件处理逻辑");
47        }
48  
49    }
50}

 

 

  WatcherProcess类:

 

 

 

 

    使用了线程安全的Hashtable来处理一次改变触发两次事件的问题,要注意的是在实际项目使用中,在通过监控文件事情触发时开一个线程WatcherProcess去处理自己业务逻辑的时候,不管业务逻辑成功或者失败(例如有异常抛出一定要try一下)一定要让WatcherProcess Completed也就是MyFileSystemWatherWatcherProcess_OnCompleted执行去移除对应变化文件的Hashtablekey,不然下次此文件改变时是无法触发你的业务逻辑的。

    

     还有就是在进行文件监控的时候, 被监控文件在写入的时候,是会有I/O冲突的,即使写入文件是FileShare.Read的也会出现,要真正解决貌似只有FileMaping方法,但是我的项目中文本的写入软件不是我们能控制的,所以只有用处理异常的方法来解决。

      各位有什么好的建议可以留言大家讨论讨论

 

 

001using System;
002using System.Collections;
003using System.IO;
004using System.Threading;
005  
006namespace test
007{
008  
009    public delegate void Completed(string key);
010  
011    public class MyFileSystemWather
012    {
013        private FileSystemWatcher fsWather;
014  
015        private Hashtable hstbWather;
016  
017        public event RenamedEventHandler OnRenamed;
018        public event FileSystemEventHandler OnChanged;
019        public event FileSystemEventHandler OnCreated;
020        public event FileSystemEventHandler OnDeleted;
021  
022        /// <summary> 
023        /// 构造函数 
024        /// </summary> 
025        /// <param name="path">要监控的路径</param> 
026        public MyFileSystemWather(string path, string filter)
027        {
028            if (!Directory.Exists(path))
029            {
030                throw new Exception("找不到路径:" + path);
031            }
032  
033            hstbWather = new Hashtable();
034  
035            fsWather = new FileSystemWatcher(path);
036            // 是否监控子目录
037            fsWather.IncludeSubdirectories = false;
038            fsWather.Filter = filter;
039            fsWather.Renamed += new RenamedEventHandler(fsWather_Renamed);
040            fsWather.Changed += new FileSystemEventHandler(fsWather_Changed);
041            fsWather.Created += new FileSystemEventHandler(fsWather_Created);
042            fsWather.Deleted += new FileSystemEventHandler(fsWather_Deleted);
043        }
044  
045        /// <summary> 
046        /// 开始监控 
047        /// </summary> 
048        public void Start()
049        {
050            fsWather.EnableRaisingEvents = true;
051        }
052  
053        /// <summary> 
054        /// 停止监控 
055        /// </summary> 
056        public void Stop()
057        {
058            fsWather.EnableRaisingEvents = false;
059        }
060  
061        /// <summary> 
062        /// filesystemWatcher 本身的事件通知处理过程 
063        /// </summary> 
064        /// <param name="sender"></param> 
065        /// <param name="e"></param> 
066        private void fsWather_Renamed(object sender, RenamedEventArgs e)
067        {
068            lock (hstbWather)
069            {
070                hstbWather.Add(e.FullPath, e);
071            }
072  
073            WatcherProcess watcherProcess = new WatcherProcess(sender, e);
074            watcherProcess.OnCompleted += new Completed(WatcherProcess_OnCompleted);
075            watcherProcess.OnRenamed += new RenamedEventHandler(WatcherProcess_OnRenamed);
076            Thread thread = new Thread(watcherProcess.Process);
077            thread.Start();
078        }
079  
080        private void WatcherProcess_OnRenamed(object sender, RenamedEventArgs e)
081        {
082            OnRenamed(sender, e);
083        }
084  
085        private void fsWather_Created(object sender, FileSystemEventArgs e)
086        {
087            lock (hstbWather)
088            {
089                hstbWather.Add(e.FullPath, e);
090            }
091            WatcherProcess watcherProcess = new WatcherProcess(sender, e);
092            watcherProcess.OnCompleted += new Completed(WatcherProcess_OnCompleted);
093            watcherProcess.OnCreated += new FileSystemEventHandler(WatcherProcess_OnCreated);
094            Thread threadDeal = new Thread(watcherProcess.Process);
095            threadDeal.Start();
096        }
097  
098        private void WatcherProcess_OnCreated(object sender, FileSystemEventArgs e)
099        {
100            OnCreated(sender, e);
101        }
102  
103        private void fsWather_Deleted(object sender, FileSystemEventArgs e)
104        {
105            lock (hstbWather)
106            {
107                hstbWather.Add(e.FullPath, e);
108            }
109            WatcherProcess watcherProcess = new WatcherProcess(sender, e);
110            watcherProcess.OnCompleted += new Completed(WatcherProcess_OnCompleted);
111            watcherProcess.OnDeleted += new FileSystemEventHandler(WatcherProcess_OnDeleted);
112            Thread tdDeal = new Thread(watcherProcess.Process);
113            tdDeal.Start();
114        }
115  
116        private void WatcherProcess_OnDeleted(object sender, FileSystemEventArgs e)
117        {
118            OnDeleted(sender, e);
119        }
120  
121        private void fsWather_Changed(object sender, FileSystemEventArgs e)
122        {
123            if (e.ChangeType == WatcherChangeTypes.Changed)
124            {
125                if (hstbWather.ContainsKey(e.FullPath))
126                {
127                    WatcherChangeTypes oldType = ((FileSystemEventArgs)hstbWather[e.FullPath]).ChangeType;
128                    if (oldType == WatcherChangeTypes.Created || oldType == WatcherChangeTypes.Changed)
129                    {
130                        return;
131                    }
132                }
133            }
134  
135            lock (hstbWather)
136            {
137                hstbWather.Add(e.FullPath, e);
138            }
139            WatcherProcess watcherProcess = new WatcherProcess(sender, e);
140            watcherProcess.OnCompleted += new Completed(WatcherProcess_OnCompleted);
141            watcherProcess.OnChanged += new FileSystemEventHandler(WatcherProcess_OnChanged);
142            Thread thread = new Thread(watcherProcess.Process);
143            thread.Start();
144        }
145  
146        private void WatcherProcess_OnChanged(object sender, FileSystemEventArgs e)
147        {
148            OnChanged(sender, e);
149        }
150  
151        public void WatcherProcess_OnCompleted(string key)
152        {
153            lock (hstbWather)
154            {
155                hstbWather.Remove(key);
156            }
157        }
158    }
159}

01using System.IO;
02  
03namespace test
04{
05    public class WatcherProcess
06    {
07        private object sender;
08        private object eParam;
09  
10        public event RenamedEventHandler OnRenamed;
11        public event FileSystemEventHandler OnChanged;
12        public event FileSystemEventHandler OnCreated;
13        public event FileSystemEventHandler OnDeleted;
14        public event Completed OnCompleted;
15  
16        public WatcherProcess(object sender, object eParam)
17        {
18            this.sender = sender;
19            this.eParam = eParam;
20        }
21  
22        public void Process()
23        {
24            if (eParam.GetType() == typeof(RenamedEventArgs))
25            {
26                OnRenamed(sender, (RenamedEventArgs)eParam);
27                OnCompleted(((RenamedEventArgs)eParam).FullPath);
28            }
29            else
30            {
31                FileSystemEventArgs e = (FileSystemEventArgs)eParam;
32                if (e.ChangeType == WatcherChangeTypes.Created)
33                {
34                    OnCreated(sender, e);
35                    OnCompleted(e.FullPath);
36                }
37                else if (e.ChangeType == WatcherChangeTypes.Changed)
38                {
39                    OnChanged(sender, e);
40                    OnCompleted(e.FullPath);
41                }
42                else if (e.ChangeType == WatcherChangeTypes.Deleted)
43                {
44                    OnDeleted(sender, e);
45                    OnCompleted(e.FullPath);
46                }
47                else
48                {
49                    OnCompleted(e.FullPath);
50                }
51            }
52        }
53    }
54}

 

 

 

 

 MyFileSystemWather类:

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值