php无法原生实时监控文件系统,linux下推荐inotify扩展(cli环境),web环境只能轮询scandir(),或通过inotifywait/powershell等外部工具协同实现,需注意权限与资源限制。

PHP 本身没有内置的、持续运行的文件系统监控能力,inotify(Linux)或 ReadDirectoryChangesW(Windows)这类底层事件监听必须依赖外部扩展或系统工具——纯 PHP 脚本靠轮询 scandir() 或 filemtime() 实现“提醒”,本质是低效模拟,不适合实时场景。
用 inotify 扩展监听目录变更(Linux 推荐)
这是最接近“实时提醒”的方案,需服务器启用 inotify 扩展(PHP ≥ 5.3.0,常见于 CLI 环境):
-
inotify_init()创建监听句柄 -
inotify_add_watch()注册目标目录,指定事件类型如IN_CREATE、IN_DELETE、IN_MOVED_TO - 用
stream_select()阻塞等待事件,避免 CPU 空转 - 收到事件后解析
inotify_read()返回的数组,提取name和mask
注意:该扩展不支持 Web SAPI(如 Apache mod_php),只能用于命令行脚本长期运行;若进程被 kill 或超时退出,监控即中断。
Web 环境下用轮询模拟“活动通知”
在 Apache/Nginx + PHP-FPM 场景中,每次 HTTP 请求都是独立生命周期,无法维持监听状态。可行做法是:
立即学习“PHP免费学习笔记(深入)”;
- 前端定时发起 AJAX 请求到一个 PHP 接口(如
/api/watch-folder.php) - 接口内调用
scandir()获取当前文件列表,与上一次结果比对(可存于file_get_contents('last_state.json')) - 仅返回差异项(新增/删除/修改时间变化超过阈值的文件)
- 为减少 IO 压力,避免检查整个目录树,限定深度 1 层,跳过
.、..和隐藏文件
缺点明显:延迟取决于轮询间隔(通常 2–10 秒),且并发高时易触发 opendir() 失败或 Too many open files 错误。
用 shell 命令 + PHP 触发通知(跨平台兼容)
绕过 PHP 自身限制,交由系统工具完成监听,PHP 只做“响应者”:
- Linux 下用
inotifywait -m -e create,delete,move_to /path/to/dir启动长进程,输出事件到管道 - 用
proc_open()在 PHP 中启动该命令,读取 stdout 流式处理 - Windows 下可用
PowerShell的Register-ObjectEvent监听FileSystemWatcher,再通过命名管道或临时文件传递信号给 PHP - 收到事件后,PHP 可调用
mail()、写入数据库、或向 WebSocket 服务推送消息
关键点:proc_open() 启动的子进程必须设置 bypass_shell=false 并正确处理信号,否则容易僵死;inotifywait 默认不递归,需加 -r 参数才监控子目录。
真正稳定的文件夹监控从来不是单靠 PHP 实现的,它要么依赖扩展(Linux CLI)、要么依赖外部工具(inotifywait / PowerShell)、要么接受轮询妥协(Web 环境)。最容易被忽略的是权限问题:www-data 用户可能无权读取目标目录,或 inotify 实例数达到 /proc/sys/fs/inotify/max_user_watches 上限导致添加失败——这些错误不会抛出 PHP 异常,只会静默失效。











