欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

PHP版自动生成文章摘要

程序员文章站 2022-08-03 13:34:26
自动生成文章摘要[javascript 版本]。 我们在写blog这样的程序时经常需要显示文章前一部分的,但是又怕不恰当的截断破坏封闭标签以造成整 个文档结构破坏...
自动生成文章摘要[javascript 版本]
我们在写blog这样的程序时经常需要显示文章前一部分的,但是又怕不恰当的截断破坏封闭标签以造成整
个文档结构破坏,使用我的函数可以在要求不高的情况下解决这个问题。

大家应该考虑这个函数在服务端应用还是在客户端应用。因为我考虑这个函数可能运行起来比较费机器,
所以安全性要求不高的情况下可以放在客户端上。

最好数据表中单独一个字段放这个摘要,这样相应的数据库查询也优化了。牺牲一点点空间换很多时间还
是划算的。

再聊一下安全性问题,主要是内容安全性。如果客户端意图更改正常的摘要信息的话,一般都是blog的主
人才有这个权力,那么他使得摘要和原文的一致性破坏就是他自己的事了。内容以外的安全性都可以在服
务端解决。所以还是推荐在客户端使用本函数。

核心代码:
最近应用了一下,发现上面的函数对多字节字符集支持得不好,因此重写了一下。

如果遇到问题,不妨试试下面的函数。

function generate_brief($text){
    global $briefing_length;
    mb_regex_encoding("utf-8");
    if(mb_strlen($text) <= brief_length ) return $text;    
    $foremost = mb_substr($text, 0, brief_length);
    $re = "<(\/?)
(p|div|h1|h2|h3|h4|h5|h6|address|pre|table|tr|td|th|input|select|textarea|object|a|ul|ol|li|
base|meta|link|hr|br|param|img|area|input|span)[^>]*(>?)";
    $single = "/base|meta|link|hr|br|param|img|area|input|br/i";    

    $stack = array(); $posstack = array();

    mb_ereg_search_init($foremost, $re, 'i');

    while($pos = mb_ereg_search_pos()){
        $match = mb_ereg_search_getregs();
        /*    [child-matching formulation]:

            $matche[1] : a "/" charactor indicating whether current "<...>" friction is 
closing part
            $matche[2] : element name.
            $matche[3] : right > of a "<...>" friction    
        */
        if($match[1]==""){
            $elem = $match[2];
            if(mb_eregi($single, $elem) && $match[3] !=""){
                continue;
            }
            array_push($stack, mb_strtoupper($elem));
            array_push($posstack, $pos[0]);            
        }else{
            $stacktop = $stack[count($stack)-1];
            $end = mb_strtoupper($match[2]);
            if(strcasecmp($stacktop,$end)==0){
                array_pop($stack);
                array_pop($posstack);
                if($match[3] ==""){
                    $foremost = $foremost.">";
                }
            }
        }
    }

    $cutpos = array_shift($posstack) - 1;    
    $foremost =  mb_substr($foremost,0,$cutpos,"utf-8");
    return $foremost;
};欢迎大家找错误。谢谢。