- 推荐学习文档
背景
我们总有这样的担忧:总有刁民想害朕,总有人偷偷在目录下删改文件,高危操作想第一时间了解,怎么办? 而且通常我们还有这样的需求:
- 监听一个目录中所有文件,文件大小到一定阀值,则处理;
- 监控某个目录,当有文件新增,立马处理;
- 监控某个目录或文件,当有文件被修改或者删除,立马能感知,进行处理;
怎么解决呢?通常来说有三个办法:
- 第一种:当事人主动通知你,这是侵入式的,需要当事人修改这部分代码来支持,依赖于当事人的自觉;(ps.显然这不太可能)
- 第二种:轮询观察,这个是无侵入式的,你可以自己写个轮询程序,每隔一段时间唤醒一次,对文件和目录做各种判断,从而得到这个目录的变化;(每次轮询累不累?不实时、不直观)
- 第三种:操作系统支持,以事件的方式通知到订阅这个事件的用户,达到及时处理的目的;
很显然第三种相对来说是最好的,具体怎么操作呢?
- 既然是操作系统的支持,那么就应该用到系统调用。系统调用直接使用略微复杂了些,Golang显然已经有了支持,使用fsnotify库就可以轻松实现这个功能。
- fsnotify库就是封装了系统调用,用来监控文件事件的。当指定目录或者文件,发生了创建,删除,修改,重命名的事件,里面就能得到通知。
Golang 的 fsnotify 的使用
- 使用方法非常简单:
- 先用 fsnotify 创建一个监听器;
- 然后放到一个单独的 goroutine 监听事件即可,通过 channel 的方式 或者 启动新的 goroutine 进行事件传递;
package main
import (
"github.com/fsnotify/fsnotify"
log "github.com/sirupsen/logrus"
)
func TestFileMonitor() {
// 创建文件/目录监听器
watcher, err := fsnotify.NewWatcher()
if err != nil {
log.Fatal("watcher new err: [%+v]", err)
return
}
defer watcher.Close()
done := make(chan bool)
go func() {
for {
select {
case event, ok := <-watcher.Events:
if !ok {
return
}
// 打印监听事件
log.Print("event:", event)