zoukankan      html  css  js  c++  java
  • 最长回文字符串计算

    定义:如果一个字符串正着读和反着读是一样的,那它就是回文串

       例如:aba 、 abba 不限制奇数偶数

    判断是否是回文字符串

      思路:从最前后各去一个做对比,如果相同则进行下一个,直到相挨为止

    function palindrome($str) {
    
        $i = 0;
        $j = strlen($str)-1;
    
        $flag = 1;
    
        while($i<$j)
        {
    
            if($str[$i] != $str[$j])
            {
                $flag = 0;
                break;
            }
    
            $i++;
            $j--;
    
        }
        
        return $flag;
        
    }

    计算最长回文字串

    方法1:

      暴力破解:计算出每个可能的字串,然后对比   

      时间复杂度:n(n^3)

    function findLongestPalindrome($str)
    {
        $palindrome = array();
        
        for($i=0;$i<strlen($str);$i++)
        {
            for($j=$i+1;$j<strlen($str);$j++)
            {
                $sub = substr($str, $i, $j-$i+1);
                if(palindrome($sub))
                {
                    $palindrome[] = strlen($sub);
                }
            }
        }
        
        return $palindrome ? max($palindrome) : 0;
        
    }

    方法2:中心扩展

    中心扩展就是把给定的字符串的每一个字母当做中心,向两边扩展,这样来找最长的子回文串。算法复杂度为O(N^2)。
    但是要考虑两种情况:
    1、像aba,这样长度为奇数。
    2、想abba,这样长度为偶数。

    function findLongestPalindrome($str)
    {
        $maxLen = 0;
        
        for($i=0;$i<strlen($str);$i++)
        {
            //奇数需要减1
            $left  = $i-(strlen($str)%2 == 0 ? 0 : 1);//左边走
            $right = $i+1;//右边走
            
            while($left>=0 && $right<strlen($str) && $str[$left] == $str[$right])
            {
                if($right-$left+1>$maxLen)
                {
                    $maxLen = $right-$left+1;
                }
                
                $left --;
                $right ++;
                
            }
        }
        
        return $maxLen;
        
    }

    方法3:Manacher

      1、奇偶数的问题,在字符串两边和中间分别插入特殊符号(如:#) 后,没个字符串都为奇数(奇数+偶数=奇数)

      2、重复访问问题,根据回文字符串的对称性快速扩展回文串

    function findLongestPalindrome($str)
    {
        //预处理,字符中间和两边增加#字符
        $strArr = str_split($str);    
        $str = '#'.implode('#', $strArr).'#';
        
        $maxLen = 0;
        
        $rl = array();//记录以该字符为中心的最长回文字符串半径
        $maxRight = 0;//现在访问到的最右边的字符
        $pos = 0; //$maxRight位置对应的轴位置
        
        for($i=0;$i<strlen($str);$i++)
        {
            if($i<$maxRight) //在左边
            {
                //根据回文串的对称性得出
                //应该为$pos-($pos-$i)],但最大不能超过 $maxRight-1 
                $rl[$i] = min($rl[$pos-($i-$pos)], $maxRight-1);
            }else{
                $rl[$i] = 1;
            }
            
            //向两边扩展 两边不能越界,且新的两边的字符还必须相等
            while($i-$rl[$i]>=0 && $i+$rl[$i]<strlen($str) && ($str[$i-$rl[$i]] == $str[$i+$rl[$i]]))
            {
                $rl[$i]++;
                
            }
            
            //更新 maxright,pos
            if($i+$rl[$i]-1>$maxRight)
            {
                $maxRight = $i+$rl[$i]-1;
                $pos = $i;
            }
            
            $maxLen = max($maxLen, $rl[$i]);
        }
        
        return $maxLen - 1;
    }

    参考:http://www.cnblogs.com/leoin2012/p/3984997.html

        https://segmentfault.com/a/1190000003914228

  • 相关阅读:
    第十二周总结
    第十一周课程总结
    第十周第十周课程总结
    第九周课程总结&实验报告(七)
    第八周课程总结&实验报告(六)
    第七周课程总结&实验报告(五)
    第六周&java实验报告四
    第五周课程总结&试验报告(三)
    学期总结
    十四周总结
  • 原文地址:https://www.cnblogs.com/siqi/p/6481672.html
Copyright © 2011-2022 走看看