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

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











