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 win下Socket方式发邮件类
Aug 21 PHP
php urlencode()与urldecode()函数字符编码原理详解
Dec 06 PHP
浅析51个PHP处理字符串的函数
Aug 02 PHP
php漏洞之跨网站请求伪造与防止伪造方法
Aug 15 PHP
PHP中ob_start函数的使用说明
Nov 11 PHP
php根据年月获取季度的方法
Mar 31 PHP
ThinkPHP模板引擎之导入资源文件方法详解
Jun 18 PHP
举例讲解PHP面对对象编程的多态
Aug 12 PHP
PHP中的session安全吗?
Jan 22 PHP
微信支付开发订单查询实例
Jul 12 PHP
laravel-admin的图片删除实例
Sep 30 PHP
YII2框架中添加自定义模块的方法实例分析
Mar 18 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
多重?l件?合查?(一)
2006/10/09 PHP
PHP个人网站架设连环讲(三)
2006/10/09 PHP
PHP基于双向链表与排序操作实现的会员排名功能示例
2017/12/26 PHP
Laravel 简单实现Ajax滚动加载示例
2019/10/22 PHP
javascript 图片裁剪技巧解读
2012/11/15 Javascript
jquery 循环显示div的示例代码
2013/10/18 Javascript
JQuery实现绚丽的横向下拉菜单
2013/12/19 Javascript
javascript生成不重复的随机数
2015/07/17 Javascript
jQuery EasyUi实战教程之布局篇
2016/01/26 Javascript
javascript中sort排序实例详解
2016/07/24 Javascript
JS获取checkbox的个数简单实例
2016/08/19 Javascript
JavaScript基于对象去除数组重复项的方法
2016/10/09 Javascript
Bootstrap源码解读排版(1)
2016/12/23 Javascript
jQuery模拟窗口抖动效果
2017/03/15 Javascript
微信小程序 蓝牙的实现实例代码
2017/06/27 Javascript
Chrome调试折腾记之JS断点调试技巧
2017/09/11 Javascript
解决Vue 浏览器后退无法触发beforeRouteLeave的问题
2017/12/24 Javascript
AngularJS实现与后台服务器进行交互的示例讲解
2018/08/13 Javascript
vue中rem的配置的方法示例
2018/08/30 Javascript
详解Vue 如何监听Array的变化
2019/06/06 Javascript
微信小程序清空输入框信息与实现屏幕往上滚动的示例代码
2020/06/23 Javascript
利用python代码写的12306订票代码
2015/12/20 Python
python getopt详解及简单实例
2016/12/30 Python
Python 多线程Threading初学教程
2017/08/22 Python
python中pika模块问题的深入探究
2018/10/13 Python
Python3中关于cookie的创建与保存
2018/10/21 Python
python实现坦克大战游戏 附详细注释
2020/03/27 Python
解决pyPdf和pyPdf2在合并pdf时出现异常的问题
2020/04/03 Python
Pytest如何使用skip跳过执行测试
2020/08/13 Python
HTML5里autofocus自动聚焦属性使用介绍
2016/06/22 HTML / CSS
Tuckernuck官网:经典的美国品质服装、鞋子和配饰
2021/01/11 全球购物
旅游业大学生创业计划书
2014/01/31 职场文书
干部竞争上岗演讲稿
2014/09/11 职场文书
2015年仓库管理员工作总结
2015/04/21 职场文书
个人维稳承诺书
2015/05/04 职场文书
仙境之桥观后感
2015/06/16 职场文书