本文主要描述如何通过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类,并传入需要监控的目录路径,以及是否制定监控的文件类型(文章前面有所介绍)。_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,代码如下:/// <summary>

/// 文件监控类,用于监控指定目录下文件以及文件夹的变化

/// </summary>

public class FileWatcher

{

private FileSystemWatcher _watcher = null;

private string _path = string.Empty;

private string _filter = string.Empty;

private bool _isWatch = false;

private CustomQueue<FileChangeInformation> _queue = null;

/// <summary>

/// 监控是否正在运行

/// </summary>

public bool IsWatch

{

get

{

return _isWatch;

}

}

/// <summary>

/// 文件变更信息队列

/// </summary>

public CustomQueue<FileChangeInformation> FileChangeQueue

{

get

{

return _queue;

}

}

/// <summary>

/// 初始化FileWatcher类

/// </summary>

/// <param name="path">监控路径</param>

public FileWatcher(string path)

{

_path = path;

_queue = new CustomQueue<FileChangeInformation>();

}

/// <summary>

/// 初始化FileWatcher类,并指定是否持久化文件变更消息

/// </summary>

/// <param name="path">监控路径</param>

/// <param name="isPersistence">是否持久化变更消息</param>

/// <param name="persistenceFilePath">持久化保存路径</param>

public FileWatcher(string path, bool isPersistence, string persistenceFilePath)

{

_path = path;

_queue = new CustomQueue<FileChangeInformation>(isPersistence, persistenceFilePath);

}

/// <summary>

/// 初始化FileWatcher类,并指定是否监控指定类型文件

/// </summary>

/// <param name="path">监控路径</param>

/// <param name="filter">指定类型文件,格式如:*.txt,*.doc,*.rar</param>

public FileWatcher(string path, string filter)

{

_path = path;

_filter = filter;

_queue = new CustomQueue<FileChangeInformation>();

}

/// <summary>

/// 初始化FileWatcher类,并指定是否监控指定类型文件,是否持久化文件变更消息

/// </summary>

/// <param name="path">监控路径</param>

/// <param name="filter">指定类型文件,格式如:*.txt,*.doc,*.rar</param>

/// <param name="isPersistence">是否持久化变更消息</param>

/// <param name="persistenceFilePath">持久化保存路径</param>

public FileWatcher(string path, string filter, bool isPersistence, string persistenceFilePath)

{

_path = path;

_filter = filter;

_queue = new CustomQueue<FileChangeInformation>(isPersistence, persistenceFilePath);

}

/// <summary>

/// 打开文件监听器

/// </summary>

public void Open()

{

if (!Directory.Exists(_path))

{

Directory.CreateDirectory(_path);

}

if (string.IsNullOrEmpty(_filter))

{

_watcher = new FileSystemWatcher(_path);

}

else

{

_watcher = new FileSystemWatcher(_path, _filter);

}

//注册监听事件

_watcher.Created += new FileSystemEventHandler(OnProcess);

_watcher.Changed += new FileSystemEventHandler(OnProcess);

_watcher.Deleted += new FileSystemEventHandler(OnProcess);

_watcher.Renamed += new RenamedEventHandler(OnFileRenamed);

_watcher.IncludeSubdirectories = true;

_watcher.EnableRaisingEvents = true;

_isWatch = true;

}

/// <summary>

/// 关闭监听器

/// </summary>

public void Close()

{

_isWatch = false;

_watcher.Created -= new FileSystemEventHandler(OnProcess);

_watcher.Changed -= new FileSystemEventHandler(OnProcess);

_watcher.Deleted -= new FileSystemEventHandler(OnProcess);

_watcher.Renamed -= new RenamedEventHandler(OnFileRenamed);

_watcher.EnableRaisingEvents = false;

_watcher = null;

}

/// <summary>

/// 获取一条文件变更消息

/// </summary>

/// <returns></returns>

public FileChangeInformation Get()

{

FileChangeInformation info = null;

if (_queue.Count > 0)

{

lock (_queue)

{

info = _queue.Dequeue();

}

}

return info;

}

/// <summary>

/// 监听事件触发的方法

/// </summary>

/// <param name="sender"></param>

/// <param name="e"></param>

private void OnProcess(object sender, FileSystemEventArgs e)

{

try

{

FileChangeType changeType = FileChangeType.Unknow;

if (e.ChangeType == WatcherChangeTypes.Created)

{

if (File.GetAttributes(e.FullPath) == FileAttributes.Directory)

{

changeType = FileChangeType.NewFolder;

}

else

{

changeType = FileChangeType.NewFile;

}

}

else if (e.ChangeType == WatcherChangeTypes.Changed)

{

//部分文件创建时同样触发文件变化事件,此时记录变化操作没有意义

//如果

if (_queue.SelectAll(

delegate(FileChangeInformation fcm)

{

return fcm.NewPath == e.FullPath && fcm.ChangeType == FileChangeType.Change;

}).Count<FileChangeInformation>() > 0)

{

return;

}

//文件夹的变化,只针对创建,重命名和删除动作,修改不做任何操作。

//因为文件夹下任何变化同样会触发文件的修改操作,没有任何意义.

if (File.GetAttributes(e.FullPath) == FileAttributes.Directory)

{

return;

}

changeType = FileChangeType.Change;

}

else if (e.ChangeType == WatcherChangeTypes.Deleted)

{

changeType = FileChangeType.Delete;

}

//创建消息,并压入队列中

FileChangeInformation info = new FileChangeInformation(Guid.NewGuid().ToString(), changeType, e.FullPath, e.FullPath, e.Name, e.Name);

_queue.Enqueue(info);

}

catch

{

Close();

}

}

/// <summary>

/// 文件或目录重命名时触发的事件

/// </summary>

/// <param name="sender"></param>

/// <param name="e"></param>

private void OnFileRenamed(object sender, RenamedEventArgs e)

{

try

{

//创建消息,并压入队列中

FileChangeInformation info = new FileChangeInformation(Guid.NewGuid().ToString(), FileChangeType.Rename, e.OldFullPath, e.FullPath, e.OldName, e.Name);

_queue.Enqueue(info);

}

catch

{

Close();

}

}

}

复制代码最后,功能调用如下://初始化监控器

FileWatcher watcher = new FileWatcher(@"D:\");

watcher.Open();

FileChangeInformation fci = null;

//获取消息

while (true)

{

//如果IsWatch为False,则可能监控内部发生异常终止了监控,需要重新开启监控

if (watcher.IsWatch)

{

//队列顶端的变更消息

fci = watcher.Get();

//处理消息的代码

//Print(fci);

}

else

{

watcher.Open();

}

Thread.Sleep(1000);

}

复制代码该程序实现了对文件目录下所有子目录和子文件的变化进行监控,并可通过FileChangeQueue属性访问文件变更消息,同时也可以设置其是否需要将数据持久化到磁盘文件中。

C#实现对文件目录的实时监控的更多相关文章

  1. 基于邮件系统的远程实时监控系统的实现 Python版

    人生苦短,我用Python~ 界内的Python宣传标语,对Python而言,这是种标榜,实际上,Python确实是当下最好用的开发语言之一. 在相继学习了C++/C#/Java之后,接触Python ...

  2. 使用gulp 合并压缩打包,实时监控文件,实现本地server

    今天不讲webpack,就说说gulp是怎么进行压缩合并打包 首先你的安装gulp : npm install gulp -g --save-dev 然后最基本的你因该知道gulp 的四个方法, gu ...

  3. 安装 log.io 实时监控 php_error 日志

    Log.io 实时监控 php_error.log 日志 开启 php_error 实时监控日志的第一步,要首先开启 php_error 的功能. vi php.ini 修改 PHP 配置文件,将 ; ...

  4. Android(Linux)实时监控串口数据

    之前在做WinCE车载方案时,曾做过一个小工具TraceMonitor,用于显示WinCE系统上应用程序的调试信息,特别是在实车调试时,用于监控和显示CAN盒与主机之间的串口数据.因为需要抢占市场先机 ...

  5. 项目-基于视频压缩的实时监控系统--tiny6410

    项目-基于视频压缩的实时监控系统--tiny6410 @国嵌linux学习笔记. 1. 构造服务端结构体 server struct server { int epfd; //保存epoll指针 st ...

  6. 实时监控log文件

    一个进程在运行,并在不断的写log,你需要实时监控log文件的更新(一般是debug时用),怎么办,不断的打开,关闭文件吗? 不用,至少有两个方法,来自两个很常用的命令: tail -f log.tx ...

  7. 利用ngxtop实时监控nginx的访问情况

    关于对nginx web server的实时访问的实时监控问题,我很久之前就想实现的,现在虽有nginx自带的status扩展,但那是全局的,无法细分到vhost,并且提供的metric也很少,加之目 ...

  8. OVM-V1.2 版发布,新增实时监控、支持一键升级

    OVM是国内首款.完全免费.企业级--混合虚拟化管理平台,OVM是从中小企业目前的困境得到启发,完全基于国内企业特点开发,更多的关注国内中小企业用户的产品需求. OVM-V1.2 该版本功能变动如下: ...

  9. Spotlight实时监控Windows Server 2008

    Windows Server 2008作为服务器平台已逐渐被推广和应用,丰富的功能和良好的稳定性为其赢得了不错的口碑.但是和Windows Server 2003相比,其系统的自我监控功能并没有多大的 ...

随机推荐

  1. 360 chrome不能登录Google账户

    用过Google chrome,枫树,360 chrome国际版,今天还是回到Google chrome了. 因为除了Chrome自己,其他的都不能正常登录Google账户(偶尔偶尔一次可以,直到今天 ...

  2. 李洪强iOS开发之让您的Xcode键字如飞

    手指在键盘上飞速跳跃,终端上的代码也随着飞舞,是的这确实很酷.优秀的程序员总是这么一群人,他们不拘于现状,不固步自封,他们喜欢新奇的事,他们把自己发挥到极致. 指法攻略 放下您钟爱的鼠标吧,在前行之中 ...

  3. Spring transaction事务之roll back回滚

    转载自:http://blog.csdn.net/lovejavaydj/article/details/7635848 试验方法: 写一个单元测试,调用一个service层方法(发生对数据库进行写操 ...

  4. java socket知识点

    3.用线程池实现TCP服务器端时,首先创建一个ServerSocket实例,然后创建N个线程,每个线程反复循环,从(共享的)ServerSocket实例接收客户端连接.当多个线程同时调用一个Serve ...

  5. Java学习笔记之:Java String类

    一.引言 字符串广泛应用在Java编程中,在Java中字符串属于对象,Java提供了String类来创建和操作字符串. 创建字符串最简单的方式如下: String str= "Hello w ...

  6. Linux Shell脚本入门:tee命令

    用途说明   在执行Linux命令时,我们可以把输出重定向到文件中,比如 ls >a.txt,这时我们就不能看到输出了,如果我们既想把输出保存到文件中,又想在屏幕上看到输出内容,就可以使用tee ...

  7. Java API —— 反射

    1.类加载器     1)类的加载         · 当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载,连接,初始化三步来实现对这个类进行初始化.         · 加载 :就是 ...

  8. JS代码片段:日期格式化

    Date.prototype.format = function(format) { var date = { "M+": this.getMonth() + 1, "d ...

  9. 简单的算法题, Find Minimum in Rotated Sorted Array 的Python实现。

    简单的算法题, Find Minimum in Rotated Sorted Array 的Python实现. 题目: Suppose a sorted array is rotated at som ...

  10. proc_dir_entry

    struct proc_dir_entry {        unsigned int low_ino;        unsigned short namelen;        const cha ...