【BZOJ】【3790】神奇项链

Manacher算法/DP



  找出所有的回文串,看做是一个个线段,那么问题就转化成了用最少的线段将整个区间覆盖起来,可以重叠,那么这就是一个DP了= =

  Orz ZKY大爷,让蒟蒻开眼界了……头一次知道原来树状数组还可以反过来用0.0

 1 /**************************************************************
 2     Problem: 3790
 3     User: Tunix
 4     Language: C++
 5     Result: Accepted
 6     Time:176 ms
 7     Memory:3716 kb
 8 ****************************************************************/
 9
10 //BZOJ 3790
11 #include<vector>
12 #include<cstdio>
13 #include<cstring>
14 #include<cstdlib>
15 #include<iostream>
16 #include<algorithm>
17 #define rep(i,n) for(int i=0;i<n;++i)
18 #define F(i,j,n) for(int i=j;i<=n;++i)
19 #define D(i,j,n) for(int i=j;i>=n;--i)
20 #define pb push_back
21 #define mk make_pair
22 #define fi first
23 #define se second
24 using namespace std;
25 typedef long long LL;
26 typedef pair<int,int> pii;
27 inline int getint(){
28     int r=1,v=0; char ch=getchar();
29     for(;!isdigit(ch);ch=getchar()) if(ch==‘-‘)r=-1;
30     for(; isdigit(ch);ch=getchar()) v=v*10+ch-‘0‘;
31     return r*v;
32 }
33 const int N=1e5+10,INF=~0u>>2;
34 /*******************template********************/
35 char s[N];
36 int a[N],p[N],f[N],d[N],n,m,size=0;
37 pii seg[N];
38 void get_seg(int l,int r){
39     if(l>r)return;
40     seg[++size]=mk(l,r);
41 }
42 bool cmp(pii a,pii b){
43     return a.se==b.se ? a.fi<b.fi : a.se<b.se;
44 }
45 void update(int x,int v){
46     for(;x;x-=x&-x) d[x]=min(d[x],v);
47 }
48 int query(int x){
49     int r=d[x];
50     for(;x && x<=size;x+=x&-x) r=min(d[x],r);
51     return r;
52 }
53 int main(){
54     while(scanf("%s",s)!=EOF){
55         memset(d,0x7f,sizeof d); d[0]=0;
56         memset(a,0,sizeof a);
57         memset(f,0,sizeof f);
58         m=n=strlen(s);
59         F(i,1,n) a[i<<1]=s[i-1];
60         n=n<<1|1; size=0;
61         int id=0,mx=0;
62         F(i,1,n){
63             if(mx>i) p[i]=min(p[2*id-i],mx-i);
64             else p[i]=0;
65             while(i-p[i]-1>0 && i+p[i]+1<=n && a[i-p[i]-1]==a[i+p[i]+1])p[i]++;
66             get_seg((i-p[i]+1)/2,(i+p[i]-1)/2);
67             if (p[i]+i>mx) mx=p[i]+i,id=i;
68         }
69         sort(seg+1,seg+size+1,cmp);
70         int ans=INF;
71         F(i,1,size){
72             f[i]=query(seg[i].fi-1)+1;
73             if (seg[i].se==m) ans=min(ans,f[i]);
74             update(seg[i].se,f[i]);
75         }
76         printf("%d\n",ans-1);
77     }
78     return 0;
79 }

3790: 神奇项链

Time Limit: 10 Sec  Memory Limit: 64 MB
Submit: 126  Solved: 63
[Submit][Status][Discuss]

Description

母亲节就要到了,小 H 准备送给她一个特殊的项链。这个项链可以看作一个用小写字

母组成的字符串,每个小写字母表示一种颜色。为了制作这个项链,小 H
购买了两个机器。第一个机器可以生成所有形式的回文串,第二个机器可以把两个回文串连接起来,而且第二个机器还有一个特殊的性质:假如一个字符串的后缀和
一个字符串的前缀是完全相同的,那么可以将这个重复部分重叠。例如:aba和aca连接起来,可以生成串abaaca或
abaca。现在给出目标项链的样式,询问你需要使用第二个机器多少次才能生成这个特殊的项链。

Input

输入数据有多行,每行一个字符串,表示目标项链的样式。

Output

多行,每行一个答案表示最少需要使用第二个机器的次数。

Sample Input

abcdcba
abacada
abcdef

Sample Output

0
2
5

HINT

每个测试数据,输入不超过 5行

每行的字符串长度小于等于 50000

Source

[Submit][Status][Discuss]

时间: 2024-08-11 10:23:11

【BZOJ】【3790】神奇项链的相关文章

BZOJ 3790 神奇项链 hash/后缀自动机+贪心

Description 母亲节就要到了,小 H 准备送给她一个特殊的项链.这个项链可以看作一个用小写字母组成的字符串,每个小写字母表示一种颜色. 为了制作这个项链,小 H 购买了两个机器.第一个机器可以生成所有形式的回文串,第二个机器可以把两个回文串连接起来,而且第二个机器还有一个特殊的性质:假如一个字符串的后缀和一个字符串的前缀是完全相同的,那么可以将这个重复部分重叠.例如:aba和aca连接起来,可以生成串abaaca或 abaca. 现在给出目标项链的样式,询问你需要使用第二个机器多少次才

BZOJ 3790 神奇项链(manacher+DP+树状数组)

[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3790 [题目大意] 问最少用几个回文串可以构成给出串,重叠部分可以合并 [题解] 我们先用manacher处理出每个位置最长的回文串, 那么题目就转化为求最少的线段来覆盖全区间,那就是经典的dp题了, dp[i]=min(dp[j]+1)(i线段的左端点-1和j线段的右端点有交) 用树状数组优化一下即可. [代码] #include <cstdio> #include <al

BZOJ 3790 神奇项链 Hash+二分

题目大意:给出一个字符串,求出这是最少由多少个回文串组成的.回文串可以重叠. 思路:将原串中的所有回文串都统计出来,然后变成一些区间,问题就转化成了区间并的问题. CODE: #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> #define MAX 400010 #define BASE 1333 #define INF 0x3f3f3f3f using

bzoj 3790 神奇项链(Manacher,DP+BIT | 贪心)

[题意] 你可以产生一个回文串,也可以将两个串合并成一个串,问产生目标串需要的最少合并次数. [思路] Manacher求出每个位置可以向两边延伸的最长回文串. 则题目转化为有若干条线段,求最少的线段将[1..n]覆盖.贪心DP皆可上,DP需要BIT优化一下. [代码] 1 #include<set> 2 #include<cmath> 3 #include<queue> 4 #include<vector> 5 #include<cstdio>

BZOJ 3790 神奇项链 Hash+二分+树状数组

题目大意:给定一个串,问这个串最少可以由回文串拼接多少次而成(拼接可以重叠) 首先将每两个字符之间插入占位符,然后Hash+二分搞出所有极大回文串(可以用manacher,我不会) 问题转化成了给定一些区间,求最少的能覆盖整个数轴的区间 将所有区间按照某一端点排序 然后上树状数组即可 回头还是去学学manacher吧... #include <cstdio> #include <cstring> #include <iostream> #include <algo

【BZOJ-3790】神奇项链 Manacher + 树状数组(奇葩) + DP

3790: 神奇项链 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 304  Solved: 150[Submit][Status][Discuss] Description 母亲节就要到了,小 H 准备送给她一个特殊的项链.这个项链可以看作一个用小写字母组成的字符串,每个小写字母表示一种颜色.为了制作这个项链,小 H 购买了两个机器.第一个机器可以生成所有形式的回文串,第二个机器可以把两个回文串连接起来,而且第二个机器还有一个特殊的性质:假如一个

BZOJ 1006 神奇的国度(弦图的染色数)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1006 题意:给定一个弦图,求最小染色数.就是用最小数目的颜色进行染色使得任意两个相邻的节点颜色不同. 思路:(1)求出弦图的完美消除序列. (2)贪心染色.从后向前用可以用的编号最小的颜色染色.在这里因为最小染色等于最大团,我直接求的最大团.为什么最小染色等于最大团呢?最大团w(G) 是包含点数最多的团,最小染色x(G)是相邻点不同色的最小颜色个数.那么w(G)<=x(G),因为最大团

bzoj3790 神奇项链

Description 母亲节就要到了,小 H 准备送给她一个特殊的项链.这个项链可以看作一个用小写字 母组成的字符串,每个小写字母表示一种颜色.为了制作这个项链,小 H 购买了两个机器.第一个机器可以生成所有形式的回文串,第二个机器可以把两个回文串连接起来,而且第二个机器还有一个特殊的性质:假如一个字符串的后缀和一个字符串的前缀是完全相同的,那么可以将这个重复部分重叠.例如:aba和aca连接起来,可以生成串abaaca或 abaca.现在给出目标项链的样式,询问你需要使用第二个机器多少次才能

bzoj 1493: [NOI2007]项链工厂(线段树)

1493: [NOI2007]项链工厂 Time Limit: 30 Sec  Memory Limit: 64 MBSubmit: 1256  Solved: 545[Submit][Status][Discuss] Description Input 输 入文件第一行包含两个整数N, c,分别表示项链包含的珠子数目以及颜色 数目.第二行包含N 个整数,x1, x2…, xn,表示从位置1 到位置N 的珠子的颜色, 1 ≤ xi ≤ c.第三行包含一个整数Q,表示命令数目.接下来的Q 行每行一