bzoj1830: [AHOI2008]Y型项链(LCP+贪心)

1830: [AHOI2008]Y型项链

题目:传送门

简要题意:

   给出三个字符串,可以对任意字符串进行操作,每次操作都可以再其中一个字符串的末尾删除或添加一个字符,求最小操作数使得所有的字符串相同

题解:

   一开始没有看到只有三个串

   仔细想想发现会是一道水ti:

   因为只有三个串嘛,而且n才50,那就对任意两个串都跑一下LCP,找出匹配率最高的两个串,然后就可以开始真正的操作了:

   首先:对于匹配率最高的两个串,对串进行删除使它变短或者进行添加使它变长所需要的操作数是一样的(比如CATB和CATCHA)

   但是当对第三个串操作的时候影响是不同的:对于第三个串,要么什么都不用做,要么删了再加,要么直接加

   对于后面两种情况,肯定都只用在乎加了多少(那肯定越少越好),那么在前面对匹配率高的两个串进行操作时,肯定让它变短啦!

   nice双倍经验(没有搞懂AHOI的操作)

代码:

 #include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm>
using namespace std;
int len,len1,len2,len3,maxx,k;
char s1[],s2[],s3[];
void get_LCP()
{
int i=;maxx=;k=;
while(s1[i]==s2[i] && i<=len1 && i<=len2)i++;
if(maxx<i)maxx=i-,k=;
i=;
while(s1[i]==s3[i] && i<=len1 && i<=len3)i++;
if(maxx<i)maxx=i-,k=;
i=;
while(s2[i]==s3[i] && i<=len2 && i<=len3)i++;
if(maxx<i)maxx=i-,k=;
}
int main()
{
scanf("%d",&len1);scanf("%s",s1+);
scanf("%d",&len2);scanf("%s",s2+);
scanf("%d",&len3);scanf("%s",s3+);
get_LCP();
if(k==)
{
int ans=;
ans+=abs(len1-maxx)+abs(len2-maxx);int i=;
while(i<=maxx && s1[i]==s3[i])i++;
i--;
ans+=(len3-i+maxx-i);
printf("%d\n",ans);
}
if(k==)
{
int ans=;
ans+=abs(len1-maxx)+abs(len3-maxx);int i=;
while(i<=maxx && s1[i]==s2[i])i++;
i--;
ans+=(len2-i+maxx-i);
printf("%d\n",ans);
}
if(k==)
{
int ans=;
ans+=abs(len2-maxx)+abs(len3-maxx);int i=;
while(i<=maxx && s1[i]==s2[i])i++;
i--;
ans+=(len1-i+maxx-i);
printf("%d\n",ans);
}
return ;
}
上一篇:click事件和mousedown、mouseup事件


下一篇:BZOJ1830: [AHOI2008]Y型项链 & BZOJ1789: [Ahoi2008]Necklace Y型项链