日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

CodeForces - 906E Reverses(回文自动机+Palindrome Series优化dp)

發(fā)布時(shí)間:2024/4/11 编程问答 37 豆豆
生活随笔 收集整理的這篇文章主要介紹了 CodeForces - 906E Reverses(回文自动机+Palindrome Series优化dp) 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

題目鏈接:點(diǎn)擊查看

題目大意:給出兩個(gè)字符串 s 和 t,每次可以在字符串 s 中選擇數(shù)個(gè)不相交的字串進(jìn)行反轉(zhuǎn),問(wèn)最少需要反轉(zhuǎn)多少次,可以使得字符串 s 和 t 相等,輸出最小反轉(zhuǎn)次數(shù)以及方案

題目分析:一個(gè)不太平凡的轉(zhuǎn)換:

構(gòu)造字符串 ss = s[ 1 ] t[ 1 ] s[ 2 ] t[ 2 ] ... s[ n ] t[ n ],如果字符串 s 在反轉(zhuǎn)?( l , r ) 之后可以與字符串 t 匹配的話,那么新字符串 ss 中的 ( 2 * l - 1 , 2 * r ) 內(nèi)的部分屬于回文串

所以現(xiàn)在問(wèn)題轉(zhuǎn)換成了,求最小價(jià)值的偶數(shù)回文串的分割方案,有一點(diǎn)需要注意的是,在本題中長(zhǎng)度為 2 的回文串是不做貢獻(xiàn)的,需要特判一下

先考慮一下 n * n 的 dp 轉(zhuǎn)移方案:設(shè) j 為回文自動(dòng)機(jī)上的節(jié)點(diǎn)

  • 如果 len[ j ] != 2 且 len[ j ] % 2 == 0 :dp[ i ] =min( dp[ i ] , dp[ i - len[ j ] ] + 1 )
  • 如果 len[ j ] == 2 :dp[ i ] = dp[ i - 2 ]
  • 然后上回文自動(dòng)機(jī)就可以直接轉(zhuǎn)移了

    再考慮用?Palindrome Series 優(yōu)化,如果不要求輸出路徑的話完全可以這樣寫(xiě):

    for(int j=last;j>1;j=anc[j]) {g[j]=f[i-len[anc[j]]-diff[j]];if(diff[j]==diff[fail[j]])g[j]=min(g[j],g[fail[j]]);if(i%2==0)f[i]=min(f[i],f[g[j]]+1); } if(i%2==0&&s[i]==s[i-1])f[i]=min(f[i],f[i-2]);

    不過(guò)這個(gè)題目需要輸出路徑,所以在維護(hù) g 數(shù)組的時(shí)候,需要維護(hù)其下標(biāo),然后間接找到最小值進(jìn)行轉(zhuǎn)移,并且在轉(zhuǎn)移時(shí)記錄一下前驅(qū)方便輸出

    代碼:

    //#pragma GCC optimize(2) //#pragma GCC optimize("Ofast","inline","-ffast-math") //#pragma GCC target("avx,sse2,sse3,sse4,mmx") #include<iostream> #include<cstdio> #include<string> #include<ctime> #include<cmath> #include<cstring> #include<algorithm> #include<stack> #include<climits> #include<queue> #include<map> #include<set> #include<sstream> #include<cassert> #include<bitset> using namespace std;typedef long long LL;typedef unsigned long long ull;const int inf=0x3f3f3f3f;const int N=1e6+100;char s[N],t[N];int n;int g[N],f[N],pre[N];struct Palindrome_tree {int nxt[N][26];int fail[N]; // 當(dāng)前節(jié)點(diǎn)最長(zhǎng)回文后綴的節(jié)點(diǎn)int len[N]; // 當(dāng)前節(jié)點(diǎn)表示的回文串的長(zhǎng)度int cnt[N]; // 當(dāng)前節(jié)點(diǎn)回文串的個(gè)數(shù), 在getcnt后可得到全部int sed[N]; // 以當(dāng)前節(jié)點(diǎn)為后綴的回文串的個(gè)數(shù)(并不是表示第i結(jié)尾的回文串的種類(lèi)數(shù),如果要求每個(gè)點(diǎn)結(jié)尾的數(shù)的回文串個(gè)數(shù),得用last)int record[N]; //record記錄了節(jié)點(diǎn)回文串的結(jié)束位置int diff[N],anc[N];char s[N];int tot; // 節(jié)點(diǎn)個(gè)數(shù)int last; // 上一個(gè)節(jié)點(diǎn)int n;//當(dāng)前字符串的長(zhǎng)度 void newnode(){tot++;memset(nxt[tot],0,sizeof(nxt[tot]));cnt[tot]=sed[tot]=len[tot]=fail[tot]=0;}void init(){n=0;tot=-1;newnode();newnode();len[0] = 0, len[1] = -1; // 0為偶數(shù)長(zhǎng)度根, 1為奇數(shù)長(zhǎng)度根tot = 1, last = 0;fail[0] = 1;}int getfail(int x, int n){while (s[n - len[x] - 1] != s[n]||n-len[x]-1<0) // 比較x節(jié)點(diǎn)回文串新建兩端是否相等//n-len[x]-1<0這個(gè)是我自己加的,多組的時(shí)候光第一個(gè)條件是不夠的,所以有錯(cuò)請(qǐng)手動(dòng)刪除x = fail[x]; // 若不同, 再比較x后綴回文串兩端return x;}void insert(char ch){int c = ch - 'a';//全小寫(xiě)要用a 全大寫(xiě)要用A 不然會(huì)錯(cuò)s[++n]=ch;int p = getfail(last, n);// 得到第i個(gè)字符可以加到哪個(gè)節(jié)點(diǎn)的兩端形成回文串if (!nxt[p][c]){newnode();len[tot] = len[p] + 2; // 在p節(jié)點(diǎn)兩端添加兩個(gè)字符fail[tot] = nxt[getfail(fail[p], n)][c]; //tot點(diǎn)的后綴回文,可以由上一個(gè)節(jié)點(diǎn)的后綴回文嘗試得到sed[tot] = sed[fail[tot]] + 1; // 以當(dāng)前節(jié)點(diǎn)為結(jié)尾的回文串個(gè)數(shù)nxt[p][c] = tot; // 新建節(jié)點(diǎn)diff[tot]=len[tot]-len[fail[tot]];anc[tot]=diff[tot]==diff[fail[tot]]?anc[fail[tot]]:fail[tot];}last = nxt[p][c]; // 當(dāng)前節(jié)點(diǎn)成為上一個(gè)節(jié)點(diǎn)cnt[last]++; //當(dāng)前節(jié)點(diǎn)回文串++record[last] = n;trans(n);}void trans(int i){for(int j=last;j>1;j=anc[j]){g[j]=i-len[anc[j]]-diff[j];if(diff[j]==diff[fail[j]]&&f[g[j]]>f[g[fail[j]]])g[j]=g[fail[j]];if(i%2==0&&f[i]>f[g[j]]+1){f[i]=f[g[j]]+1;pre[i]=g[j];}}if(i%2==0&&s[i]==s[i-1]&&f[i]>=f[i-2]){f[i]=f[i-2];pre[i]=i-2;}}void get_cnt(){for (int i = tot; i > 0; i--)cnt[fail[i]] += cnt[i];//fail[i] 的節(jié)點(diǎn) 為 i 節(jié)點(diǎn)的后綴回文串, 所以個(gè)數(shù)相加} }tree; int main() { #ifndef ONLINE_JUDGE // freopen("data.in.txt","r",stdin); // freopen("data.out.txt","w",stdout); #endif // ios::sync_with_stdio(false);memset(f,inf,sizeof(f));memset(g,inf,sizeof(g));f[0]=0;tree.init();scanf("%s%s",s+1,t+1);n=strlen(s+1);for(int i=1;i<=n;i++){tree.insert(s[i]);tree.insert(t[i]);}if(f[n<<1]==inf)puts("-1");else{printf("%d\n",f[n<<1]);int pos=n<<1;while(pos){if(pos-pre[pos]>2)printf("%d %d\n",pre[pos]/2+1,pos/2);pos=pre[pos];}}return 0; }

    ?

    總結(jié)

    以上是生活随笔為你收集整理的CodeForces - 906E Reverses(回文自动机+Palindrome Series优化dp)的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

    如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

    主站蜘蛛池模板: 亚洲网友自拍 | 精品人妻一区二区三区日产 | 波多野结衣一区二区三区高清 | 日韩中文av在线 | 婷婷人体 | 91一区二区在线观看 | 亚洲中文字幕在线观看 | 中国一级特黄毛片 | 女人被狂躁c到高潮喷水电影 | 国产偷怕 | 少妇床戏av | 末发成年娇小性xxxxx | 亚洲一区二区在线免费观看 | 黄色大片免费网站 | 久久伊人草| 久久久久69 | 男女xx网站 | 日韩伦理一区 | av一级二级| 久久久久久欧美精品se一二三四 | 最近国语视频在线观看免费播放 | 天堂在线亚洲 | 韩国久久久久 | 超碰人人超碰 | 神马午夜伦理 | 麻豆网站免费看 | 中文字幕在线观看亚洲 | 网站av| 免费无码不卡视频在线观看 | 亚洲精品乱码久久 | 黄色成人小视频 | 在线污视频| 特黄做受又粗又大又硬老头 | 国产在线一区视频 | 黄色成人av在线 | 午夜8888 | 偷看洗澡一二三区美女 | 欣赏asian国模裸体pics | 国产调教在线观看 | 天天色天天看 | 国产高潮视频 | ass亚洲肉体欣赏pics | 国语对白真实视频播放 | 色呦呦网站在线观看 | 日本一区二区视频在线播放 | 亚洲乱强伦 | 91久久精品一区二区三 | 欧美高清另类 | 美女自拍视频 | 欧美亚洲一二三区 | 欧美黄色一级视频 | 福利综合网| 午夜电影你懂的 | 国产啊v在线 | 国产黄色自拍 | 爱欲av | 脱美女衣服亲摸揉视频 | 成人免费观看av | 国产精品第一区 | 国产av无码专区亚洲av麻豆 | 波多野结衣一区二区三区在线 | 精品少妇一二三区 | 免费看毛片的网站 | 三级全黄做爰在线观看 | 日韩三级在线观看 | 青草99| 男人视频网站 | youjizz少妇| 国产精品免费精品一区 | 国产精品人成在线观看免费 | 婷婷激情四射 | 羞羞羞网站 | 吃奶摸下的激烈视频 | 成人学院中文字幕 | 182tv午夜 | 亚洲乱亚洲乱妇 | 国产福利精品视频 | 成人3d动漫一区二区三区 | 午夜电影天堂 | 欧美性爱精品在线 | 日本一区二区三区在线观看视频 | 久久77 | 少妇精品高潮欲妇又嫩中文字幕 | 天天夜夜操| 日韩网站在线观看 | 日本人妻一区 | 无遮挡裸光屁屁打屁股男男 | 呦呦网 | 日韩精品专区 | 久久久久极品 | 老司机精品视频在线 | 午夜性福利 | 青青草福利视频 | 视频精品久久 | 欧美亚洲不卡 | 欧美a∨ | 成年视频在线观看 | 免费在线视频一区二区 | 亚洲一区二区三区在线播放 |