博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
【Atcoder】CODE FESTIVAL 2017 qual C D - Yet Another Palindrome Partitioning
阅读量:4620 次
发布时间:2019-06-09

本文共 939 字,大约阅读时间需要 3 分钟。

【题意】

给定只含小写字母的字符串,要求分割成若干段使段内字母重组顺序后能得到回文串,求最少分割段数。n<=2*10^5

【题解】

关键在于快速判断一个字符子串是否合法,容易发现合法仅当不存在或只存在一个奇数字符,其余字符均为偶数。

当涉及到奇偶性(%2)时,很自然能想到异或。

将小写字母a~z转化2^0~2^25,那么一个字符子串合法当且仅当其连续异或值是0或2^i(0<=i<=25)

令f[i]表示前i个合法的最少段数,sum[i]表示异或前缀和,则有:

f[i]=min(f[j])+1,sum[i]^sum[j]=0||2^i,也就是在前面所有合法的j中取最小的f[j]。

将合法条件移项,得到sum[i]^(0||2^i)=sum[j],那么对于当前的i,可以快速算出需要的sum[j]。

而sum值只有2*10^5个,可以用map存起来,然后就可以快速取用。

或者sum值本身不大,根据题目空间直接开数组也没问题。

复杂度O(26*n)。

#include
using namespace std;const int maxn = 200000 + 50;char a[maxn];int f[maxn], sum[1 << 26];int main(){ scanf("%s", a + 1); int n = strlen(a + 1); memset(sum, 0x3f, sizeof(sum)); int g = 0; f[0] = 0; sum[0] = 0; for (int i = 1; i <= n; ++i){ int x = 1 << (a[i] - 'a'); g ^= x; f[i] = sum[g]; for (int j = 0; j < 26; ++j) f[i] = min(sum[g ^ (1 << j)], f[i]); ++ f[i]; sum[g] = min(sum[g], f[i]); } cout << f[n] << endl; return 0;}

转载于:https://www.cnblogs.com/juruohx/p/7783677.html

你可能感兴趣的文章
IntelliJ IDEA 把java项目导出成可执行的jar
查看>>
DynamicReports
查看>>
鼠标经过图像改变实现
查看>>
二分查找法
查看>>
Spring3升级到Spring4时, 运行时出现找不到MappingJacksonHttpMessageConverter的情况
查看>>
详解缓冲区溢出攻击以及防范方法
查看>>
分布式事务解决方案(一) 2阶段提交 & 3阶段提交 & TCC
查看>>
android之网格布局和线性布局实现注册页面
查看>>
BZOJ 1014: [JSOI2008]火星人prefix( splay + hash )
查看>>
安装ejabberd2并配置MySQL为其数据库
查看>>
angular repeat
查看>>
android 图片圆角化控件
查看>>
java第三次作业
查看>>
HP Jack介绍
查看>>
敏捷软件开发(3)---COMMAND 模式 & Active Object 模式
查看>>
poj 1062 昂贵的聘礼 解题报告
查看>>
get the page name from url
查看>>
visual studio中csproj文件中的project guid改为小写 ( notepad++ 正则)
查看>>
TeeChart显示三维的图形,使用Surface
查看>>
如何使用 Idea 远程调试 Java 代码
查看>>