题目链接

题意描述

给定一个字符串 \(s\),你初始拥有一个空串 \(t\),每次可以选择这个字符串的一个 Border,去掉它后接在 \(t\) 的后面,操作后 \(s\) 不变,给出一个上限 \(w\),求出在 \([1,w]\) 中有多少长度可以被拼出。

题目分析

首先可以看出,抛开字符串,把每个可以拼上去的长度筛选出来,问题就转化为有 \(k\) 个不同的数,通过加法操作可以表示出范围内的多少个数,这显然是一个同余最短路的问题。

模仿通常同余最短路的操作,从每个点延伸 \(k\) 条边后跑一边最短路即可,但是这样只能得到 30 分,原因在于边数过大。

我们考虑优化,既然边这么多,能不能去掉一些没有用的部分,反过来考虑字符串 Border 的一些性质,注意到周期好像是个很有用的部分,能把冗余部分压缩,我们试图从这方面考虑。

对于周期和 Border,我们有一个很显然的性质:每个长度大于 \(\frac{|s|}{2}\) 的 Border 剩下那部分如果长度是 \(|s|\) 的因子,那么它是原字符串的一个周期,而任意一个周期与一个最短周期,记为 \(p\),\(q\)(\(p>q\)),\(\gcd(p,q)\) 也是原字符串的一个周期。

因为 \(p\) 和 \(q\) 的最大公因数也是一个周期,而 \(\gcd(p,q)\leq\min(p,q)\),从上述定义可以看出 \(q=\gcd(p,q)\),因此 \(q|p\),当 \(p\) 所对应的 Border 取最小的时,即 \(p\) 最大,我们就可以发现 \(q\) 复制后只要不超过 \(p\),一定对应一个 Border,而这些 Border 刚好组成一个等差数列

因为所有的周期都能由最小周期组成,所以所有大于 \(\frac{|s|}{2}\) 的 Border 都构成一个等差数列,而把小于的最大的 Border 提出来变成一个新字符串继续递归,可以得到最多不超过 \(\log(n)\) 条等差数列。

这个性质很有用,我们可以把每条数列单独拉出来跑同余最短路,优化边数即可达到目的。

具体实现

把最小的当成模数,每个点就只有一条出边,整个图总共会形成 \(\gcd(a,d)\)个简单环(\(a\) 为 最小值,\(d\),为公差)。

接着把环分离,从最小值开始转圈跑最短路,最后跑下一条等差数列的时候改变一下模数即可(如果从小到大跑只用把每个最小值所在位置,改成模新模数的位置,并查看前面会不会对后面增加的同余类产生影响即可)。

一共有 \(\log(n)\) 条数列,每条数列跑最短路复杂度为 \(O(n)\),整体时间为 $ O(n \log n)$。

代码

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<set>
#include<cstring>
#include<queue>
#define N 500005
using namespace std;
typedef unsigned long long ll;
ll cnt,bor[N<<3],nex[N<<3];
ll res[N<<3];
ll dis[N<<3],lim;
struct Node{
ll num,val;
};
ll top,Q[N<<3],q2[N<<3];
char s[N];
ll gcd(ll x,ll y){
if(!y) return x;
else return gcd(y,x%y);
}
void get_border(ll n){
ll ans=0;
for(ll i=2;i<=n;i++){
while(ans && s[ans+1]!=s[i]) ans=nex[ans];
if(s[ans+1]==s[i]) ans++;
nex[i]=ans;
} while(ans){bor[++cnt]=n-ans;ans=nex[ans];} bor[++cnt]=n;
}
void rfs(ll len){
ll siz=gcd(lim,len);
for(ll i=0;i<lim;i++)res[i]=dis[i];
for(ll i=0;i<len;i++)dis[i]=0x7f7f7f7f7f7f7f7f;
for(ll i=0;i<lim;i++){
ll tmp=res[i]%len;
dis[tmp]=min(dis[tmp],res[i]);
}
for(ll i=0;i<siz;i++){
top=0;Q[++top]=i;ll tmp=(i+lim)%len;
while(tmp!=i){Q[++top]=tmp;tmp=(tmp+lim)%len;}
for(ll j=top+1;j<=top*2;j++) Q[j]=Q[j-top];
for(ll j=2;j<=(top<<1);j++) dis[Q[j]]=min(dis[Q[j]],dis[Q[j-1]]+lim);
}
lim=len; }
void update(ll a,ll d,ll num){
if(d<0) return;
ll siz=gcd(lim,d);
for(ll i=0;i<siz;i++){
top=0;ll tmp=(i+d)%lim;Q[++top]=i;
while(tmp!=i){Q[++top]=tmp;tmp=(tmp+d)%lim;}
tmp=1;
for(ll j=2;j<=top;j++) if(dis[Q[j]]<dis[Q[tmp]]) tmp=j;
for(ll j=1;j<=top;j++) res[j]=Q[j];
for(ll j=tmp;j<=top;j++) Q[j-tmp+1]=res[j];
for(ll j=1;j<tmp;j++) {Q[j+(top-tmp+1)]=res[j];}
ll head=1,tail=1;q2[1]=1;
for(ll j=2;j<=top;j++){ while(head<=tail && j-q2[head]>num) head++; dis[Q[j]]=min(dis[Q[j]],dis[Q[q2[head]]]+a+d*(j-q2[head])); while(tail>=head && dis[Q[q2[tail]]]+d*(j-q2[head])>dis[Q[j]]) tail--;
q2[++tail]=j; }
}
}
ll solve(ll n,ll w){
ll L=1;
while(L<=cnt){
ll R=L;
while(bor[R+1]-bor[R]==bor[L+1]-bor[L]) R++;
rfs(bor[L]);
if(R>cnt) {L=R;continue;}
update(bor[L],(L==R?0:bor[L+1]-bor[L]),R-L);
L=R;
}
ll ans=0;
for(ll i=0;i<lim;i++){
if(dis[i]>w) continue;
ans+=(w-dis[i])/lim+1;
}
return ans;
}
int main(){
ll t;
cin>>t;
while(t--){
ll n,w;
cin>>n>>w>>(s+1);if(w<n){cout<<0<<endl;continue;}cnt=0;w-=n;
lim=n;dis[0]=0;
get_border(n);
cout<<solve(n,w)<<endl;
for(ll i=0;i<n;i++) dis[i]=0x7f7f7f7f7f7f7f7f;
}
}

P4156 [WC2016]论战捆竹竿 题解的更多相关文章

  1. luogu P4156 [WC2016]论战捆竹竿

    传送门 官方题解(证明都在这) 神仙题鸭qwq 转化模型,发现这题本质就是一个集合,每次可以加上集合里的数,问可以拼出多少不同的数 首先暴力需要膜意义下的最短路,例题戳这 然后这个暴力可以优化成N^2 ...

  2. Luogu4156 WC2016 论战捆竹竿 KMP、同余类最短路、背包、单调队列

    传送门 豪华升级版同余类最短路-- 官方题解 主要写几个小trick: \(1.O(nm)\)实现同余类最短路: 设某一条边长度为\(x\),那么我们选择一个点,在同余类上不断跳\(x\),可以形成一 ...

  3. bzoj4406: [Wc2016]论战捆竹竿&&uoj#172. 【WC2016】论战捆竹竿

    第二次在bzoj跑进前十竟然是因为在UOJ卡常致死 首先这个题其实就是一个无限背包 一般做法是同余最短路,就是bzoj2118: 墨墨的等式可以拿到30分的好成绩 背包是个卷积就分治FFT优化那么下面 ...

  4. BZOJ4406 WC2016 论战捆竹竿

    Problem BZOJ Solution 显然是一个同余系最短路问题,转移方案就是所有|S|-border的长度,有 \(O(n)\) 种,暴力跑dijkstra的复杂度为 \(O(n^2\log ...

  5. 「WC2016」论战捆竹竿

    「WC2016」论战捆竹竿 前置知识 参考资料:<论战捆竹竿解题报告-王鉴浩>,<字符串算法选讲-金策>. Border&Period 若前缀 \(pre(s,x)​\ ...

  6. UOJ#172. 【WC2016】论战捆竹竿 字符串 KMP 动态规划 单调队列 背包

    原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ172.html 题解 首先,这个问题显然是个背包问题. 然后,可以证明:一个字符串的 border 长度可 ...

  7. 【WC2016】论战捆竹竿

    已经快三周了啊--终于把挖的坑填了-- 首先显然是把除了自身的所有border拿出来,即做 \(\left\{ n - b_1, n - b_2, \dots, n - b_k, n \right\} ...

  8. UOJ#172. 【WC2016】论战捆竹竿

    传送门 首先这个题目显然就是先求出所有的 \(border\),问题转化成一个可行性背包的问题 一个方法就是同余类最短路,裸跑 \(30\) 分,加优化 \(50\) 分 首先有个性质 \(borde ...

  9. 【LuoguP4156】论战捆竹竿

    题目链接 题意简述 你有一个长度为 n 的字符串 , 将它复制任意次 , 复制出的串的前缀可以与之前的串的后缀重叠在一起 , 问最后总共可能的长度数目 , 长度不能超过 \(w\) 多组数据. \(n ...

  10. Loj 507 接竹竿 题解

    Loj链接:接竹竿 $ {\scr \color {SkyBlue}{\text{Solution}}} $ 题目大意: 给定一个数组,每次加入一种颜色的数,可以取走与它颜色相同的两个数之间的所有数, ...

随机推荐

  1. 2023ccpc大学生程序设计竞赛-wh

    对于大一的我,只听说线下大型比赛,而第一次参加也必然心情激动,生为大一,由于没有参赛经历,所有不知道参赛技巧,所以三个人像个无头苍蝇一样,跟着榜单做,我作为写码的,其他两名队友负责思路和想法,第一道签 ...

  2. Redis 主从同步原理

    一.什么是主从同步? 主从同步,就是将数据冗余备份,主库(Master)将自己库中的数据,同步给从库(Slave). 从库可以一个,也可以多个,如图所示: 二.为什么需要主从同步? Redis 虽然有 ...

  3. python:时间模块dateutil

    安装 pip install python-dateutil dateutil模块主要有两个函数,parser和rrule. 其中parser是根据字符串解析成datetime,而rrule则是根据定 ...

  4. Wampserver64 报错:无法启动此程序,因为计算机中丢失 MSVCR110.dll。尝试重新安装该程序以解决此问题。

    缺少环境配置, 程序下载地址如下: https://www.microsoft.com/zh-cn/download/confirmation.aspx?id=30679 点击下载,下载完成后,双击程 ...

  5. CoaXPress 2.0 FPGA HOST IP Core Linux Demo

    目录 Hello-FPGA CoaXPress 2.0 Host FPGA IP Core Linux Demo 4 1 说明 4 2 设备连接 7 3 VIVADO FPGA工程 7 4 调试说明 ...

  6. 《高级程序员 面试攻略 》RocketMQ 如何保证顺序性

    RocketMQ 提供了一种称为顺序消息的机制来确保消息的顺序性.下面是一些关键的方法和概念: 1. 顺序消息:顺序消息是指在发送和消费过程中,消息按照特定的顺序进行处理.RocketMQ 通过将消息 ...

  7. DDD 架构分层,MQ消息要放到那一层处理?

    作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 本文的宗旨在于通过简单干净实践的方式教会读者,使用 Docker 配置 RocketMQ 并在 ...

  8. Docker容器怎么安装Vim编辑器

    ​ 在现代软件开发和系统管理中,Docker已经成为一个不可或缺的工具.它允许我们轻松地创建.部署和运行应用程序,以及构建可移植的容器化环境.然而,在Docker容器中安装特定的工具可能会有一些挑战, ...

  9. 【Hexo】配置主流搜索引擎收录流程记录

    目录 是否已经被收录 生成站点地图 提交站点地图 Google 注册 Search Console 验证网站所有权 提交站点地图 Bing 从 GSC 导入 手动添加网站 手动请求编入索引 参考资料 ...

  10. 《SQL与数据库基础》03. SQL-DML

    目录 DML 数据插入 数据删除 数据更新 本文以 MySQL 为例 DML 数据插入 给指定字段添加数据: INSERT INTO 表(字段1, 字段2, ......, 字段n) VALUES(值 ...