• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            zoj2744_Palindromes

            Posted on 2010-10-13 11:00 李東亮 閱讀(1632) 評(píng)論(0)  編輯 收藏 引用

            ZOJ 2744 Palindromes

            這道題要求出一個(gè)長(zhǎng)度不大于5000的字符串所有子串中回文字符串的個(gè)數(shù)。可以用暴力brute-force方法解決,但是會(huì)TLE。上網(wǎng)搜索了一下大家的解決方案,主要有兩種:DP法和分治法。仔細(xì)消化了一下,感覺(jué)受益匪淺。

            首先說(shuō)下DP方法,其實(shí)碰到這種具有很多重復(fù)計(jì)算的子問(wèn)題的問(wèn)題,很容易想到動(dòng)態(tài)規(guī)劃DP。對(duì)于這個(gè)問(wèn)題可以這樣理解:每次都從下標(biāo)0開(kāi)始搜索長(zhǎng)度依次為23、……、n的子串,長(zhǎng)度為1的子串肯定是回文。所以可以設(shè)立數(shù)組bool dp[n][n],其中dp[i][i]=truedp[i][j]表示字符串中從ij的子串是否是回文,這樣對(duì)于上面的搜索過(guò)程,要判斷str[i-j]要是否回文,可以分為兩種情況:

            1.      ij之間沒(méi)有其它字符,所以只要str[i]==str[j]即可。

            2.      ij之間還有其它的字符,則要求str[i]==str[j]并且dp[i+1][j-1]=true。這也很好理解,從i+1j-1的子串是回文,再兩端加上相同的字符,肯定也是回文。

            參考代碼如下:

            #include <cstdio>
            #include 
            <cstdlib>
            #include 
            <iostream>
            #include 
            <cstring>
            using namespace std;

            char s[5001];
            bool dp[5001][5001];

            int main(void)
            {
                
            int i, j;
                
            int len;
                
            int cnt;
                
            int k;
                
            while(gets(s))
                {
                    len 
            = strlen(s);
                    cnt 
            = len;
                    
            for (i = 0; i < len; ++i)
                    {
                        dp[i][i] 
            = 0;
                    }
                    
            for (i = 1; i < len; ++i)
                    {
                        
            for (j = 0; j < len-i; ++j)
                        {
                            k 
            = i+j;
                            dp[j][k] 
            = false;
                            
            if (s[j] == s[k])
                            {
                                
            if (j+1 < k-1)
                                {
                                    
            if (dp[j+1][k-1])
                                    {
                                        dp[j][k] 
            = true;
                                        
            ++cnt;
                                    }
                                }
                                
            else
                                {
                                    dp[j][k] 
            = true;
                                    
            ++cnt;
                                }
                            }
                        }
                    }
                    printf(
            "%d\n", cnt);
                }
                
            return 0;
            }

            第二種思路是分治法,對(duì)于每個(gè)字符可以以它為中心向兩端擴(kuò)展,依次判斷,這樣同樣避免了很多重復(fù)判斷。在擴(kuò)展過(guò)程中,如果擴(kuò)展到當(dāng)前情況發(fā)現(xiàn)不是回文,這可以停止擴(kuò)展。

            擴(kuò)展時(shí)需要分兩種情況:

            1.奇數(shù)子串?dāng)U展。例如對(duì)于字符串ababa,從a[2]=a開(kāi)始擴(kuò)展,奇數(shù)擴(kuò)展會(huì)考慮a兩邊的b,是回文,繼續(xù)向外擴(kuò)展。

            2.偶數(shù)子串?dāng)U展。對(duì)于同樣的字符串,從a[2]=a開(kāi)始擴(kuò)展 ,都屬擴(kuò)展會(huì)依次考慮子串a[1-2]=ba不是回文,停止擴(kuò)展。

            參考代碼如下:

            #include<stdio.h>
            #include
            <string.h>
            int count=0;
            char s[5001];   
            int main(){
                   
            int k,l,r;
                   
            int len;
                   
            while(gets(s)){
                          count
            =0;               
                          len
            =strlen(s);        
                          
            for(k=1;k<len-1;k++){  
                                 l
            =k-1,r=k+1;
                                 
            while(l>=0&&r<len){  //奇數(shù)長(zhǎng)度串
                                        if(s[l--]==s[r++])
                                               count
            ++;
                                        
            else
                                               
            break;
                                 }
                                 l
            =k-1,r=k;
                                 
            while(l>=0&&r<len){   //偶數(shù)長(zhǎng)度串
                                        if(s[l--]==s[r++])
                                               count
            ++;
                                        
            else
                                               
            break;
                                 }
                                
                          }
                          
            if(s[len-1]==s[len-2])  //偶數(shù)長(zhǎng)度的串少算了最后兩個(gè)字符,補(bǔ)上
                                 count++;
                          count
            +=len;   
                          printf(
            "%d\n",count);
                   }    
                   
            return 0;
            }



            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            posts - 12, comments - 1, trackbacks - 0, articles - 1

            Copyright © 李東亮

            无码任你躁久久久久久老妇App| 久久影院亚洲一区| 99久久无码一区人妻a黑| 国产精品国色综合久久| 久久精品国产色蜜蜜麻豆 | 久久久久久久波多野结衣高潮| 欧美久久久久久| 久久久中文字幕| 久久婷婷色综合一区二区| 2021少妇久久久久久久久久| 久久久久综合国产欧美一区二区| 午夜精品久久久内射近拍高清| 奇米综合四色77777久久| 久久久精品视频免费观看| 无码人妻久久一区二区三区免费| 香蕉久久一区二区不卡无毒影院 | 国产V综合V亚洲欧美久久| 久久精品国产亚洲7777| 97久久香蕉国产线看观看| 中文字幕久久亚洲一区| 久久精品国产影库免费看| 99久久无色码中文字幕人妻| 国产精品美女久久久免费 | 久久嫩草影院免费看夜色| 99久久婷婷免费国产综合精品| 亚洲国产成人乱码精品女人久久久不卡| 亚洲综合伊人久久大杳蕉| 伊人久久成人成综合网222| 91性高湖久久久久| 国产精品久久久久…| 热re99久久精品国99热| 无码人妻少妇久久中文字幕蜜桃| 久久影院久久香蕉国产线看观看| 一本色道久久88加勒比—综合| 久久免费精品视频| 国产精品美女久久久久| 99国产精品久久久久久久成人热| 亚洲精品无码久久久影院相关影片| 亚洲Av无码国产情品久久| 伊人热热久久原色播放www| 久久久国产精品|