PHP 修复未正常关闭的HTML标签实现代码(支持嵌套和就近闭合)


Posted in PHP onJune 07, 2012

fixHtmlTag
version 0.2
这个版本解决了上次遗留的问题,即就近闭合和嵌套闭合问题。具体可以看代码的注释。

<?php /** 
* fixHtmlTag 
* 
* HTML标签修复函数,此函数可以修复未正确闭合的 HTML 标签 
* 
* 由于不确定性因素太多,暂时提供两种模式“嵌套闭合模式”和 
* “就近闭合模式”,应该够用了。 
* 
* 这两种模式是我为了解释清楚此函数的实现而创造的两个名词, 
* 只需明白什么意思就行。 
* 1,嵌套闭合模式,NEST,为默认的闭合方式。即 "<body><div>你好" 
* 这样的 html 代码会被修改为 "<body><div>你好</div></body>" 
* 2,就近闭合模式,CLOSE,这种模式会将形如 "<p>你好<p>为什么没有 
* 闭合呢" 的代码修改为 "<p>你好</p><p>为什么没有闭合呢</p>" 
* 
* 在嵌套闭合模式(默认,无需特殊传参)下,可以传入需要就近闭合的 
* 标签名,通过这种方式将类似 "<body><p>你好</p><p>我也好" 转换为 
* "<body><p>你好</p><p>我也好</p></body>"的形式。 
* 传参时索引需要按照如下方式写,不需要修改的设置可以省略 
* 
* $param = array( 
* 'html' => '', //必填 
* 'options' => array( 
* 'tagArray' => array(); 
* 'type' => 'NEST', 
* 'length' => null, 
* 'lowerTag' => TRUE, 
* 'XHtmlFix' => TRUE, 
* ) 
* ); 
* fixHtmlTag($param); 
* 
* 上面索引对应的值含义如下 
* string $html 需要修改的 html 代码 
* array $tagArray 当为嵌套模式时,需要就近闭合的标签数组 
* string $type 模式名,目前支持 NEST 和 CLOSE 两种模式,如果设置为 CLOSE,将会忽略参数 $tagArray 的设置,而全部就近闭合所有标签 
* ini $length 如果希望截断一定长度,可以在此赋值,此长度指的是字符串长度 
* bool $lowerTag 是否将代码中的标签全部转换为小写,默认为 TRUE 
* bool $XHtmlFix 是否处理不符合 XHTML 规范的标签,即将 <br> 转换为 <br /> 
* 
* @author IT不倒翁 <itbudaoweng@gmail.com> 
* @version 0.2 
* @link http://yungbo.com IT不倒翁 
* @link http://enenba.com/?post=19 某某 
* @param array $param 数组参数,需要赋予特定的索引 
* @return string $result 经过处理后的 html 代码 
* @since 2012-04-14 
*/ 
function fixHtmlTag($param = array()) { 
//参数的默认值 
$html = ''; 
$tagArray = array(); 
$type = 'NEST'; 
$length = null; 
$lowerTag = TRUE; 
$XHtmlFix = TRUE; 
//首先获取一维数组,即 $html 和 $options (如果提供了参数) 
extract($param); 
//如果存在 options,提取相关变量 
if (isset($options)) { 
extract($options); 
} 
$result = ''; //最终要返回的 html 代码 
$tagStack = array(); //标签栈,用 array_push() 和 array_pop() 模拟实现 
$contents = array(); //用来存放 html 标签 
$len = 0; //字符串的初始长度 
//设置闭合标记 $isClosed,默认为 TRUE, 如果需要就近闭合,成功匹配开始标签后其值为 false,成功闭合后为 true 
$isClosed = true; 
//将要处理的标签全部转为小写 
$tagArray = array_map('strtolower', $tagArray); 
//“合法”的单闭合标签 
$singleTagArray = array( 
'<meta', 
'<link', 
'<base', 
'<br', 
'<hr', 
'<input', 
'<img' 
); 
//校验匹配模式 $type,默认为 NEST 模式 
$type = strtoupper($type); 
if (!in_array($type, array('NEST', 'CLOSE'))) { 
$type = 'NEST'; 
} 
//以一对 < 和 > 为分隔符,将原 html 标签和标签内的字符串放到数组中 
$contents = preg_split("/(<[^>]+?>)/si", $html, -1, PREG_SPLIT_NO_EMPTY | PREG_SPLIT_DELIM_CAPTURE); 
foreach ($contents as $tag) { 
if ('' == trim($tag)) { 
$result .= $tag; 
continue; 
} 
//匹配标准的单闭合标签,如<br /> 
if (preg_match("/<(\w+)[^\/>]*?\/>/si", $tag)) { 
$result .= $tag; 
continue; 
} 
//匹配开始标签,如果是单标签则出栈 
else if (preg_match("/<(\w+)[^\/>]*?>/si", $tag, $match)) { 
//如果上一个标签没有闭合,并且上一个标签属于就近闭合类型 
//则闭合之,上一个标签出栈 
//如果标签未闭合 
if (false === $isClosed) { 
//就近闭合模式,直接就近闭合所有的标签 
if ('CLOSE' == $type) { 
$result .= '</' . end($tagStack) . '>'; 
array_pop($tagStack); 
} 
//默认的嵌套模式,就近闭合参数提供的标签 
else { 
if (in_array(end($tagStack), $tagArray)) { 
$result .= '</' . end($tagStack) . '>'; 
array_pop($tagStack); 
} 
} 
} 
//如果参数 $lowerTag 为 TRUE 则将标签名转为小写 
$matchLower = $lowerTag == TRUE ? strtolower($match[1]) : $match[1]; 
$tag = str_replace('<' . $match[1], '<' . $matchLower, $tag); 
//开始新的标签组合 
$result .= $tag; 
array_push($tagStack, $matchLower); 
//如果属于约定的的单标签,则闭合之并出栈 
foreach ($singleTagArray as $singleTag) { 
if (stripos($tag, $singleTag) !== false) { 
if ($XHtmlFix == TRUE) { 
$tag = str_replace('>', ' />', $tag); 
} 
array_pop($tagStack); 
} 
} 
//就近闭合模式,状态变为未闭合 
if ('CLOSE' == $type) { 
$isClosed = false; 
} 
//默认的嵌套模式,如果标签位于提供的 $tagArray 里,状态改为未闭合 
else { 
if (in_array($matchLower, $tagArray)) { 
$isClosed = false; 
} 
} 
unset($matchLower); 
} 
//匹配闭合标签,如果合适则出栈 
else if (preg_match("/<\/(\w+)[^\/>]*?>/si", $tag, $match)) { 
//如果参数 $lowerTag 为 TRUE 则将标签名转为小写 
$matchLower = $lowerTag == TRUE ? strtolower($match[1]) : $match[1]; 
if (end($tagStack) == $matchLower) { 
$isClosed = true; //匹配完成,标签闭合 
$tag = str_replace('</' . $match[1], '</' . $matchLower, $tag); 
$result .= $tag; 
array_pop($tagStack); 
} 
unset($matchLower); 
} 
//匹配注释,直接连接 $result 
else if (preg_match("/<!--.*?-->/si", $tag)) { 
$result .= $tag; 
} 
//将字符串放入 $result ,顺便做下截断操作 
else { 
if (is_null($length) || $len + mb_strlen($tag) < $length) { 
$result .= $tag; 
$len += mb_strlen($tag); 
} else { 
$str = mb_substr($tag, 0, $length - $len + 1); 
$result .= $str; 
break; 
} 
} 
} 
//如果还有将栈内的未闭合的标签连接到 $result 
while (!empty($tagStack)) { 
$result .= '</' . array_pop($tagStack) . '>'; 
} 
return $result; 
}
PHP 相关文章推荐
php获取地址栏信息的代码
Oct 08 PHP
PHP array_multisort()函数的使用札记
Jul 03 PHP
php中使用Curl、socket、file_get_contents三种方法POST提交数据
Aug 12 PHP
PHP调用.NET的WebService 简单实例
Mar 27 PHP
PHP 开发者该知道的 5 个 Composer 小技巧
Feb 03 PHP
总结PHP中数值计算的注意事项
Aug 14 PHP
PHP用FTP类上传文件视频等的简单实现方法
Sep 23 PHP
php版微信自动登录并获取昵称的方法
Sep 23 PHP
php+mysql+jquery实现日历签到功能
Feb 27 PHP
PHP 访问数据库配置通用方法(json)
May 20 PHP
php利用array_search与array_column实现二维数组查找
Jul 08 PHP
laravel 模型查询按照whereIn排序的示例
Oct 16 PHP
PHP 文本文章分页代码 按标记或长度(不涉及数据库)
Jun 07 #PHP
PHP 查找字符串常用函数介绍
Jun 07 #PHP
php中通过curl smtp发送邮件
Jun 05 #PHP
Smarty的配置与高级缓存技术分享
Jun 05 #PHP
PHP Parse Error: syntax error, unexpected $end 错误的解决办法
Jun 05 #PHP
php中json_decode()和json_encode()的使用方法
Jun 04 #PHP
php安全之直接用$获取值而不$_GET 字符转义
Jun 03 #PHP
You might like
第一个无线电台是由谁发明的
2021/03/01 无线电
php抽奖小程序的实现代码
2013/06/18 PHP
PHP时间戳 strtotime()使用方法和技巧
2013/10/29 PHP
PHP后台微信支付和支付宝支付开发
2017/04/28 PHP
PHP实现生成模糊图片的方法示例
2017/12/21 PHP
Jquery CheckBox全选方法代码附js checkbox全选反选代码
2010/06/09 Javascript
js函数setTimeout延迟执行的简单介绍
2013/07/17 Javascript
javascript date格式化示例
2013/09/25 Javascript
showModalDialog在谷歌浏览器下会返回Null的解决方法
2013/11/27 Javascript
JavaScript时间转换处理函数
2015/04/14 Javascript
3个可以改善用户体验的AngularJS指令介绍
2015/06/18 Javascript
使用JS批量选中功能实现更改数据库中的status状态值(批量展示)
2016/11/22 Javascript
node.js入门教程之querystring模块的使用方法
2017/02/27 Javascript
Vue三层嵌套路由的示例代码
2018/05/05 Javascript
JavaScript设计模式之构造器模式(生成器模式)定义与用法实例分析
2018/07/26 Javascript
layui实现文件或图片上传记录
2018/08/28 Javascript
JS中封装axios来管控api的2种方式
2019/09/11 Javascript
vue input标签通用指令校验的实现
2019/11/05 Javascript
vue $set 给数据赋值的实例
2019/11/09 Javascript
javascript 模块依赖管理的本质深入详解
2020/04/30 Javascript
vue实现简易图片左右旋转,上一张,下一张组件案例
2020/07/31 Javascript
JavaScript实现通讯录功能
2020/12/27 Javascript
[01:08:24]DOTA2-DPC中国联赛 正赛 RNG vs Phoenix BO3 第一场 2月5日
2021/03/11 DOTA
CentOS 6.5下安装Python 3.5.2(与Python2并存)
2017/06/05 Python
python去除扩展名的实例讲解
2018/04/23 Python
python异步实现定时任务和周期任务的方法
2019/06/29 Python
python点击鼠标获取坐标(Graphics)
2019/08/10 Python
使用HTML5的链接预取功能(link prefetching)给网站提速
2012/12/13 HTML / CSS
牵手50香港:专为黄金岁月的单身人士而设的交友网站
2020/08/14 全球购物
见习期自我鉴定
2014/01/31 职场文书
遗嘱公证书标准样本
2014/04/08 职场文书
法人授权委托书公证范本
2014/09/14 职场文书
2014年平安建设工作总结
2014/11/19 职场文书
汉字听写大会观后感
2015/06/12 职场文书
创业计划书之寿司
2019/07/19 职场文书
Python如何用re模块实现简易tokenizer
2022/05/02 Python