国内最全IT社区平台 联系我们 | 收藏本站
华晨云阿里云优惠2
您当前位置:首页 > php开源 > php教程 > php截断带html字符串文章内容的方法

php截断带html字符串文章内容的方法

来源:程序员人生   发布时间:2013-10-26 05:12:14 阅读次数:4125次

文章截断使用主要是在列表页面时我没有写描述这样只能在文章中截取字符串了,但使用php 自带函数会导致div未结束,从而页面混乱了,那么要如何解决此问题呢?

博主写好一篇文章,博客后台一般会在搜索页面或者列表页面给出文章标题和截断了的的文章部分作为进一步阅读的入口。

Function: mb_substr( $str, $start, $length, $encoding )

$str,需要截断的字符串

$start,截断开始处

$length,长度(注意,这个跟mb_strimwidth不同,1就代表一个中文字符)

$encoding,编码,我设为 utf-8

例,截断文章标题,控制在15个文字,代码如下:

<?php echo mb_substr('www.phpfensi.com原创', 0, 15,"utf-8"); ?>

这样对于纯文本没问题,但是我的是中间有html标签的于是问题来了,怎样截断一篇文章,注意,这篇文章不仅仅是普通的字符串文本,而是包含了各种格式化标签和样式内容的文本,如果处理不当,这些闭合标签无法正常关闭,从而破坏整个文档流。

如果单纯是纯文本,下面这个函数差不多是够用的,代码如下:

  1. <?php 
  2.     /** 
  3.      * 字符串截取,支持中文和其他编码 
  4.      * 
  5.      * @param string $str 需要转换的字符串 
  6.      * @param string $start 开始位置 
  7.      * @param string $length 截取长度 
  8.      * @param string $charset 编码格式 
  9.      * @param string $suffix 截断字符串后缀 
  10.      * @return string 
  11.      */ 
  12.     function substr_ext($str$start=0, $length$charset="utf-8"$suffix=""
  13.     { 
  14.         if(function_exists("mb_substr")){ 
  15.              return mb_substr($str$start$length$charset).$suffix
  16.     } 
  17.         elseif(function_exists('iconv_substr')){ 
  18.              return iconv_substr($str,$start,$length,$charset).$suffix
  19.         } 
  20.         $re['utf-8']  = "/[x01-x7f]|[xc2-xdf][x80-xbf]|[xe0-xef][x80-xbf]{2}|[xf0-xff][x80-xbf]{3}/"
  21.         $re['gb2312'] = "/[x01-x7f]|[xb0-xf7][xa0-xfe]/"
  22.         $re['gbk']    = "/[x01-x7f]|[x81-xfe][x40-xfe]/"
  23.         $re['big5']   = "/[x01-x7f]|[x81-xfe]([x40-x7e]|xa1-xfe])/"
  24.         preg_match_all($re[$charset], $str$match); 
  25.         $slice = join("",array_slice($match[0], $start$length)); 
  26.         return $slice.$suffix
  27.     } 

但是,如果需要截断是网页中的某部分格式化文本,上面的函数就不够用了,它不具备处理格式化标签的能力。

这时,需要一个新函数,它应该是以上函数的升级加强版,它必须有能力正确的处理标签,下面找到一个

strip_tags() 函数剥去 HTML、XML 以及 PHP 的标签。

例子1,代码如下:

  1. <?php 
  2. echo strip_tags("Hello <b>world!</b>"); 
  3. ?> 

输出:Hello world!

这样就好做了我们只要在上面基础上如下操作,代码如下:

  1. <?php 
  2. $a = strip_tags("Hello <b>world!</b>"); 
  3. substr_ext( $a,10) ; 
  4. //但是发现html不见了这个也不是什么好的解决办法了。 
  5. ?> 

接着google 发现cns写了一个支持html截取字符串的函数,代码如下:

  1. /** 
  2.  
  3.  * 获取字符在字符串中第N次出现的位置 
  4.  * @param string $text 字符串 
  5.  * @param string $key 字符 
  6.  * @param int $int N 
  7.  * @return int  
  8.  */ 
  9. function strpos_int($text$key$int
  10.     $keylen = strlen($key); 
  11.     global $textlen
  12.     if (!$textlen
  13.         $textlen = strlen($text); 
  14.     static $textpos = 0; 
  15.     $pos = strpos($text$key); 
  16.     $int--; 
  17.     if ($pos
  18.     { 
  19.         if ($int == 0) 
  20.             $textpos+=$pos
  21.         else 
  22.             $textpos+=$pos + $keylen
  23.     } 
  24.     else 
  25.     { 
  26.         $int = 0; 
  27.         $textpos = $textlen
  28.     } 
  29.     if ($int > 0) 
  30.     { 
  31.         strpos_int(substr($text$pos + $keylen), $key$int); 
  32.     } 
  33.     return $textpos
  34.  
  35. /** 
  36.  * 截取HTML 
  37.  * @param string $string  HTML 字符串 
  38.  * @param int $length 截取的长度 
  39.  * @param string $dot 
  40.  * @param string $append 
  41.  * @return string 
  42.  */ 
  43. function cuthtml($string$length$dot = ' ...'$append = ""
  44.     $str = strip_tags($string);//先过滤标签 
  45.     $new_str = iconv_substr($str, 0, $length'utf-8'); 
  46.     $last = iconv_substr($new_str, -1, 1, 'utf-8'); 
  47.     $sc = substr_count($new_str$last); 
  48.     $position = strpos_int($string$last$sc); //获取截取真实的长度 
  49.     if (function_exists('tidy_parse_string'))//服务器开启tidy的话 直接用函数不全html代码即可 
  50.     { 
  51.         $options = array("show-body-only" => true); 
  52.         return tidy_parse_string(mb_substr($string, 0, $position) . $dot . $append$options'UTF8'); 
  53.     } else //没有开启tidy 
  54.     { 
  55.         if (strlen($string) <= $position
  56.         { 
  57.             return $string
  58.         } 
  59.         $pre = chr(1); 
  60.         $end = chr(1); 
  61.         $string = str_replace(array('&''"''<''>'), array($pre . '&' . $end$pre . '"' . $end$pre . '<' . $end$pre . '>' . $end), $string); 
  62.         $strcut = ''
  63.         $n = $tn = $noc = 0; 
  64.         while ($n < strlen($string)) 
  65.         { 
  66.             $t = ord($string[$n]); 
  67.             if ($t == 9 || $t == 10 || (32 <= $t && $t <= 126)) 
  68.             { 
  69.                 $tn = 1; 
  70.                 $n++; 
  71.                 $noc++; 
  72.             } elseif (194 <= $t && $t <= 223) 
  73.             { 
  74.                 $tn = 2; 
  75.                 $n += 2; 
  76.                 $noc += 2; 
  77.             } elseif (224 <= $t && $t <= 239) 
  78.             { 
  79.                 $tn = 3; 
  80.                 $n += 3; 
  81.                 $noc += 2; 
  82.             } elseif (240 <= $t && $t <= 247) 
  83.             { 
  84.                 $tn = 4; 
  85.                 $n += 4; 
  86.                 $noc += 2; 
  87.             } elseif (248 <= $t && $t <= 251) 
  88.             { 
  89.                 $tn = 5; 
  90.                 $n += 5; 
  91.                 $noc += 2; 
  92.             } elseif ($t == 252 || $t == 253) 
  93.             { 
  94.                 $tn = 6; 
  95.                 $n += 6; 
  96.                 $noc += 2; 
  97.             } else 
  98.             { 
  99.                 $n++; 
  100.             } 
  101.             if ($noc >= $position
  102.             { 
  103.                 break
  104.             } 
  105.         } 
  106.         if ($noc > $position
  107.         { 
  108.             $n -= $tn
  109.         } 
  110.         $strcut = substr($string, 0, $n); 
  111.         $strcut = str_replace(array($pre . '&' . $end$pre . '"' . $end$pre . '<' . $end$pre . '>' . $end), array('&''"''<''>'), $strcut); 
  112.         $pos = strrpos($strcutchr(1)); 
  113.         if ($pos !== false) 
  114.         { 
  115.             $strcut = substr($strcut, 0, $pos); 
  116.         } 
  117.         return $strcut . $dot . $append
  118.     } 
  119. }
生活不易,码农辛苦
如果您觉得本网站对您的学习有所帮助,可以手机扫描二维码进行捐赠
程序员人生
------分隔线----------------------------
分享到:
------分隔线----------------------------
关闭
程序员人生