TL;DR
引
后续
昨天重新部署好网站之后发现,网页的顶部跟浏览器的边框之间出现了一条很宽的缝隙,一开始以为是 CSS 出了问题,打开控制台才发现 <body> 标签里面莫名其妙的多了一个  的东西,<head> 的内容也都到了 <body> 里
百度了才发现是因为一开始图方便用 Windows 自带的记事本去编辑 PHP 文件,这东西保存是用带 BOM 的 UTF-8 (UTF-8 with BOM)编码保存的,这个 BOM 头会放到 PHP 文件的文件头里面,而 PHP 规定在 <?php ?> 外面的字符都会默认输出,从而导致页面多了一个  。另外 UTF-8 with BOM 还会导致<head> 的内容解析到 <body> 里面。
要想去掉  只需要重新用不带 DOM 的 UTF-8 (UTF-8 without BOM)编码保存就好了。
秉持着问题自产自销原则,PHP 出现的问题就应该由 PHP 自己来解决。又找到了这个
里面有一段 PHP 代码,只需要放在根目录下面访问一下就可以自动去掉所有文件的 BOM 头。不过如果文件比较多的话执行的时间会比较长,所以自己加了个判断文件类型的功能,只需要去掉 PHP 文件就好了,下面是新的代码。
<?php
header('content-Type: text/html; charset=utf-8');
if(isset($_GET['dir'])){
$basedir=$_GET['dir'];
}else{
$basedir='.';
}
$auto=1;/*设置为1标示检测BOM并去除,设置为0标示只进行BOM检测,不去除*/
echo '当前查找的目录为:'.$basedir.'当前的设置是:';
echo $auto?'检测文件BOM同时去除检测到BOM文件的BOM<br />':'只检测文件BOM不执行去除BOM操作<br />';
checkdir($basedir);
function checkdir($basedir){
$ext = pathinfo($file, PATHINFO_EXTENSION);
if($dh=opendir($basedir)){
while (($file=readdir($dh)) !== false){
if($file != '.' && $file != '..'){
if(!is_dir($basedir.'/'.$file)){
$ext = pathinfo($file, PATHINFO_EXTENSION);
if ($ext == "php") { //只检测 php 文件
echo '文件: '.$basedir.'/'.$file .checkBOM($basedir.'/'.$file).' <br>';
}
}else{
$dirname=$basedir.'/'.$file;
checkdir($dirname);
}
}
}
closedir($dh);
}
}
function checkBOM($filename){
global $auto;
$contents=file_get_contents($filename);
$charset[1]=substr($contents,0,1);
$charset[2]=substr($contents,1,1);
$charset[3]=substr($contents,2,1);
if(ord($charset[1])==239 && ord($charset[2])==187 && ord($charset[3])==191){
if($auto==1){
$rest=substr($contents,3);
rewrite($filename,$rest);
return (' <font color=red>找到BOM并已自动去除</font>');
}else{
return (' <font color=red>找到BOM</font>');
}
}else{
return (' 没有找到BOM');
}
}
function rewrite($filename,$data){
$filenum=fopen($filename,'w');
flock($filenum,LOCK_EX);
fwrite($filenum,$data);
fclose($filenum);
}
?>避免办法?不要用辣鸡记事本编辑 PHP 文件咯,不然又像我一样折腾了一晚上o(TヘTo)
