chmod 必然更新 ctime,但不影响 mtime 和 atime;若 mtime 变化,说明存在其他写操作。可通过 stat()+touch() 在 PHP 中恢复 mtime/atime,但 ctime 不可避免且不可逆。

chmod 后文件时间戳被重置?这是正常行为
PHP 的 chmod() 函数本身不修改文件内容,但会触发系统级 inode 元数据更新——其中 ctime(状态更改时间)必然变化,mtime 和 atime 则保持不变。你观察到的“时间戳变了”,大概率是误把 ctime 当成了 mtime;若确实发现 mtime 被动更新,那不是 chmod() 导致的,而是调用前/后有其他写操作(如 file_put_contents()、fopen(..., 'a') 等)。
想改权限又不扰动 mtime/atime?用 touch + chmod 组合
Linux/macOS 下可借助系统 touch 命令冻结时间戳再改权限。PHP 中需分两步执行,且必须确保目标文件未被其他进程写入:
- 先用
touch -r $file $file.touchstamp保存原始时间戳到临时文件 - 执行
chmod修改权限 - 再用
touch -r $file.touchstamp $file恢复mtime和atime - 最后删掉临时文件:
unlink($file . '.touchstamp')
注意:touch -r 在部分老旧 BusyBox 环境中不可用;Windows 不支持该语法,需换用 PowerShell 的 (Get-Item $path).LastWriteTime = $origTime 方式(PHP 中需 shell_exec() 配合)。
PHP 原生方案:用 stat() + touch() 手动还原 mtime/atime
绕过 shell 依赖,纯 PHP 实现的关键是:在 chmod() 前读取原始时间戳,之后用 touch() 显式设回。但要注意 touch() 默认只改 mtime,需传入第三个参数指定 atime:
立即学习“PHP免费学习笔记(深入)”;
$stat = stat($file); chmod($file, 0644); touch($file, $stat['mtime'], $stat['atime']);
这个方法在绝大多数 PHP 环境下有效,但有两个硬限制:touch() 无法修改 ctime(系统强制更新,不可逆),且若 PHP 进程无权修改文件访问时间(如挂载时用了 noatime),atime 将无法还原。
为什么 chmod 无法保留 ctime?这不是 bug,是 Unix 设计
ctime(change time)记录的是 inode 元数据最后一次变更的时间,包括权限、所有者、链接数等任何属性改动。只要调用 chmod(),内核就必须更新 ctime —— 这是 POSIX 强制要求,PHP 无法绕过,任何语言都一样。如果你的业务逻辑依赖 ctime 不变,说明设计上混淆了 ctime 和 mtime 的语义;真正该关注的通常是 mtime(内容最后修改时间),它在单纯改权限时本来就不会变。











