PHP rename() 替换文件名需确保源目标路径存在且可写,不自动创建目录,中文名需统一UTF-8编码并规避Windows权限、占用及跨卷限制,推荐先校验再操作。
直接用 rename() 函数就能完成文件名替换,但必须确保源路径和目标路径都存在且权限正确。它不自动创建父目录,也不处理中文编码问题。
rename() 返回 false 并报错 Warning: rename(): No such file or directory
overwrite 选项,但 PHP 原生 rename() 不支持该参数)file_exists() 和 is_writable() 检查if (file_exists('/path/old_文件.txt') && is_writable(dirname('/path/new_文件.txt'))) {
$result = rename('/path/old_文件.txt', '/path/new_文件.txt');
if (!$result) {
error_log('rename failed: ' . error_get_last()['message']);
}
}
PHP 文件系统函数(包括 rename()、file_exists()、scandir())底层调用的是操作系统 API,对 UTF-8 编码的中文文件名在不同环境表现不一致:Linux 通常没问题,Windows 默认使用 GBK/GB2312,会导致乱码或找不到文件。
mb_convert_encoding() 或 iconv() 不能直接用于路径字符串——它们可能把 UTF-8 转成 GBK 字节序列,但 PHP 函数仍按原始字节传给系统,结果不可控Chinese_China.65001 或类似)var_dump(bin2hex($filename)); 看实际字节流是否与磁盘中一致绕过编码争议的务实做法:不依赖中文文件名参与逻辑,而是用英文 ID + 数据库映射。但若必须保留中文名,需加一层防御性处理。
mb_strlen($name, 'UTF-8') > 0 确认字符串是合法 UTF-8,避免二进制污染/、\、控制字符等危险字节:preg_replace('/[\x00-\x1f\x7f\\\\\/\?:\*\|"\\^]/u', '_', $name)
dirname() + mkdir(..., 0755, true) 确保上级目录存在$safe_name = preg_replace('/[\x00-\x1f\x7f\\\\\/\?<>:\*\|"\\^]/u', '_', $original_name);
$safe_name = mb_substr($safe_name, 0, 200, 'UTF-8') . '.txt';
$target = __DIR__ . '/uploads/' . $safe_name;
$dir = dirname($target);
if (!is_dir($dir)) {
mkdir($dir, 0755, true);
}
rename($source, $target);
在 Windows 上,rename() 对中文路径失败,八成不是编码问题,而是文件被占用或权限卡死。
lsof(WSL)或 Process Explorer 查找句柄IIS_IUSRS,Apache 的 SYSTEM)对目标目录有“修改”权限,不只是“读取”C:\ 到 D:\),rename() 在 Windows 上本质是 move,跨卷会失败,需改用 copy() + unlink()
exec('cmd /c ren "' . escapeshellarg($old) . '" "' . escapeshellarg($new) . '"'),但注意 shell 注入风险,仅限可信输入ls -la 或 dir /u 看一眼真实字节,比猜编码更可靠。