PHP中explode(“/n”, $text)在Windows上失效是因为Windows用//r//n换行,导致//r残留;应先用str_replace([“//r//n”,”//r”], “//n”, $text)归一化再分割,避免空行和乱码。

PHP 中 explode("/n", $text) 为什么在 Windows 上失效?
因为 Windows 默认用 /r/n 换行,Linux/macOS 用 /n,而 explode("/n", $text) 遇到 /r/n 会把 /r 留在每行末尾,导致后续处理(比如 trim、匹配、入库)出错。这不是 PHP 的 bug,是跨平台文本换行标准不一致造成的。
常见错误现象:var_dump(explode("/n", "a/r/nb/r/nc")) 得到 ["a/r", "b/r", "c"],而非预期的 ["a", "b", "c"]。
- 别直接用
explode("/n", $text)处理用户上传或跨平台传入的文本 - 别依赖
PHP_OS_FAMILY做条件判断——文件可能来自任意系统,运行环境不等于来源环境 -
str_replace("/r/n", "/n", $text)必须放在str_replace("/r", "/n", ...)之前,否则/r/n → /n后残留的孤立/r会被二次转成/n,造成空行
统一换行符的推荐写法:先归一化再分割
核心思路是把所有换行变体(/r/n、/r、/n)全部转成单一 /n,再用 explode("/n", ...) 安全分割。
最简健壮写法:
立即学习“PHP免费学习笔记(深入)”;
$normalized = str_replace(["/r/n", "/r"], "/n", $text);
$lines = explode("/n", $normalized);
这个顺序不能反:先替 /r/n,再替 /r,避免 /r/n → /n/n 这类错误。不需要正则,str_replace 数组参数性能更好且兼容 PHP 5.6+。
- 如果需过滤空行,用
array_filter($lines, 'strlen'),别用array_filter($lines)(会误删 “0” 这样的有效行) - 若原始文本含 UTF-8 BOM,建议前置
ltrim($text, "/xEF/xBB/xBF"),否则首行可能带不可见字符 - 对超大文本(>10MB),
explode会生成巨大数组,此时应改用preg_split('//r/n|/r|/n/', $text, -1, PREG_SPLIT_NO_EMPTY)配合PREG_SPLIT_NO_EMPTY,但注意 PCRE 栈限制
用 file() 还是 explode?看场景
如果文本来自本地文件且可读取,file($path, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES) 是最省心的选择——它内部已自动处理所有换行变体,并默认去掉末尾换行符和空行。
但注意:file() 会把整个文件读入内存,不适合处理 GB 级日志;且无法处理字符串变量(比如 POST 提交的 textarea 内容),这时必须走归一化 + explode 路线。
-
file()返回的每行末尾不含换行符,explode方案需自行trim()或靠归一化保证干净 - 若需保留空行,
file()要去掉FILE_SKIP_EMPTY_LINES,而explode方案天然保留 - Windows 下用
fopen读取时若以"rt"模式(文本模式),PHP 会自动转换/r/n → /n,但此行为受auto_detect_line_endings配置影响,不推荐依赖
真正容易被忽略的点:textarea 提交和 JSON 传输的隐性换行
Web 表单中 在不同浏览器提交时,换行符可能被标准化为 /r/n(Chrome/Firefox)或保持原样(某些移动端 WebView),而通过 AJAX 发送 JSON 时,JavaScript 的 JSON.stringify 会把换行转义为 /n 字符串,后端收到的是字面量 "//n" 而非真实换行符。
所以拿到 $_POST['content'] 或 json_decode(file_get_contents('php://input')) 后,务必先检查实际换行符:
var_dump(bin2hex(substr($text, 0, 4))); // 查看前几个字节十六进制
看到 "0d0a" 就是 /r/n,"0a" 是 /n,"0d" 是旧 Mac 的 /r。别凭经验猜。
- JSON 场景下,前端若用
JSON.stringify({text: textarea.value}),后端需先stripslashes(如果 magic_quotes_gpc 开启过)再处理,现代框架通常已屏蔽该问题,但仍建议加日志验证 - 数据库存取时,MySQL 的
TEXT字段本身不关心换行符,但显示给前端时,HTML 会忽略换行,需配合nl2br()或 CSSwhite-space: pre-line
