标签:Beads power sum2 sum1 MAXN ull cnt 珍珠项链
https://loj.ac/problem/2427
题目描述
给出一段序列\(A\),求一个\(k\)使将序列\(A\)分为\(k\)段(不是倍数最后一段舍去)不同的段数最多。一个串的反转和它本身相同。
思路
这道题\(A\)的长度并不大,我们可以暴力枚举\(k\),对于每个\(k\)计算不同串的数目,再更新答案。需要注意这里并没有说一个串与它的顺序无关,二只是与它的反转有关,我们只要正反处理两次\(Hash\)值即可。不过这道题有点卡单\(Hash\),最好用双\(Hash\),不过我懒得写了,调一调\(b\)就\(A\)了。
代码
#include <bits/stdc++.h>
using namespace std;
typedef unsigned long long ull;
const ull p=10000019;
const ull MAXN=2e5+10;
ull sum1[MAXN],sum2[MAXN],power[MAXN],ans[MAXN],n,a[MAXN];
ull f_hash(ull l,ull r,bool f)
{
if(f)return sum1[r]-sum1[l-1]*power[r-l+1];
else return sum2[l]-sum2[r+1]*power[r-l+1];
}
set<ull>s;
ull cal(int k)
{
s.clear();
for(int i=1;i+k-1<=n;i+=k)
{
ull tmp=min(f_hash(i,i+k-1,1),f_hash(i,i+k-1,0));
s.insert(tmp);
}
return s.size();
}
int main()
{
scanf("%llu",&n);
for(ull i=1;i<=n;i++)
scanf("%llu",&a[i]);
power[0]=1;
for(ull i=1;i<=n;i++)
{
sum1[i]=sum1[i-1]*p+a[i];
power[i]=power[i-1]*p;
}
for(ull i=n;i>=1;i--)
sum2[i]=sum2[i+1]*p+a[i];
// for(ull i=1;i<=n;i++)
// printf("%llu %llu\n",sum1[i],sum2[i]);
ull maxx=0,cnt=0;
for(ull i=1;i<=n;i++)
{
ull now=cal(i);
if(now>maxx)
{
maxx=now;
cnt=0;
}
if(maxx==now)
ans[++cnt]=i;
}
printf("%llu %llu\n",maxx,cnt);
for(int i=1;i<=cnt;i++)
printf("%llu ",ans[i]);
return 0;
}
标签:Beads,power,sum2,sum1,MAXN,ull,cnt,珍珠项链 来源: https://www.cnblogs.com/fangbozhen/p/11767000.html
本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享; 2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关; 3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关; 4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除; 5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。