传送门

首先这个题目显然就是先求出所有的 \(border\),问题转化成一个可行性背包的问题

一个方法就是同余类最短路,裸跑 \(30\) 分,加优化 \(50\) 分

首先有个性质

\(border\) 分成的等差数列的个数不超过 \(log\)

和回文树的性质的证明类似瞎画图一下就行了

我们注意到可以一个一个等差数列的更新最短路

要做到这个,必须能从之前的等差数列的模数 \(n\) 转移到当前等差数列的 \(x\)

假设模 \(n\) 的最短路为 \(f\),模 \(x\) 的为 \(g\)

只需要 \(f_i\) 更新 \(g_{f_i~mod~x}\) 之后 \(g\) 自己通过添加 \(n\) 更新即可

现在考虑 \(g\) 每次 \(+n\) 更新

注意到把每次 \(+n\) 的下标弄出来,一定是若干个环,环之间独立

显然每个的最小值不会再次更新,那么找到这个点就变成了链,然后一个个向后 \(+n\) 更新即可

再考虑每个等差数列的内部更新,模数我们选择首项 \(x\),这样才比较可做

设公差为 \(d\),长度为 \(len\)

同样的,把每次 \(+d\) 的下标弄出来,还是若干个独立的环

找到最小值的位置编号 \(0\) 对变成的链向后一一编号

那么对于第 \(i\) 个点,可以从 \(j\) 加上 \(x+d\times(i-j)\) 转移,这样的 \(j\) 必须满足 \(i-j<len\)

这个东西显然可以单调队列优化一波

然后就可能可以通过这一题了

关于被hack这件事情,卡卡常就好了

# include <bits/stdc++.h>
using namespace std;
typedef long long ll; const int maxn(5e5 + 5); int test, n, nxt[maxn], len[maxn], cnt, vis[maxn], idx, id[maxn];
ll w, ans, f[maxn], g[maxn], inf, que[maxn];
char s[maxn];
queue <int> q; inline void Calc(int lst, int u, int d, int num) {
int i, j, k, mnp, hd, tl;
for (i = 0; i < lst; ++i) g[i] = f[i];
for (i = 0; i < u; ++i) f[i] = inf;
for (i = 0; i < lst; ++i) if (g[i] != inf) f[g[i] % u] = min(f[g[i] % u], g[i]);
++idx;
for (i = 0; i < u; ++i)
if (vis[i] != idx) {
mnp = i, vis[i] = idx;
for (j = (i + lst) % u; j ^ i; j = (j + lst) % u) {
if (f[j] < f[mnp]) mnp = j;
vis[j] = idx;
}
for (k = mnp, j = (mnp + lst) % u; j ^ mnp; k = j, j = (j + lst) % u)
f[j] = min(f[j], f[k] + lst);
}
++idx;
for (i = 0; i < u; ++i)
if (vis[i] != idx) {
mnp = i, vis[i] = idx, hd = 0, tl = -1;
for (j = (i + d) % u; j ^ i; j = (j + d) % u) {
if (f[j] < f[mnp]) mnp = j;
vis[j] = idx;
}
que[0] = f[mnp], id[++tl] = 0;
for (k = 1, j = (mnp + d) % u; j ^ mnp; j = (j + d) % u, ++k) {
while (hd <= tl && k - id[hd] >= num) ++hd;
if (hd <= tl) f[j] = min(f[j], que[hd] + u + (ll)(k - id[hd]) * d);
while (hd <= tl && que[tl] - (ll)id[tl] * d > f[j] - (ll)k * d) --tl;
que[++tl] = f[j], id[tl] = k;
}
}
} inline void Solve() {
int i, j, u, d, lst;
scanf("%d%lld", &n, &w), ans = 0;
scanf(" %s", s + 1);
for (i = 2, j = 0; i <= n; ++i) {
while (j && s[i] != s[j + 1]) j = nxt[j];
j += s[i] == s[j + 1], nxt[i] = j;
}
cnt = 0, j = n;
while (j) len[++cnt] = n - nxt[j], j = nxt[j];
sort(len + 1, len + cnt + 1), --cnt;
memset(f, 63, sizeof(f)), inf = f[0];
f[n % len[1]] = n, u = lst = n;
reverse(len + 1, len + cnt + 1);
for (i = 1; i < cnt; i = j) {
d = len[i] - len[i + 1], j = i + 1;
while (j <= cnt && len[j - 1] - len[j] == d) ++j;
u = len[j - 1], Calc(lst, u, d, j - i), lst = u;
}
if (cnt) u = len[cnt], Calc(lst, u, 0, 1);
for (i = 0; i < u; ++i) if (f[i] <= w) ans += (w - f[i]) / u + 1;
printf("%lld\n", ans);
} int main() {
scanf("%d", &test);
while (test) --test, Solve();
return 0;
}

UOJ#172. 【WC2016】论战捆竹竿的更多相关文章

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

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

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

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

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

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

  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. 【LuoguP4156】论战捆竹竿

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

  9. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

随机推荐

  1. asp.net c# 虾米音乐API

    最近用到虾米音乐的功能,主要是做一个分享音乐功能,找到好多代码,但是比较杂,有用的很少,因 此在此记录下,方便以后自己使用. 对于第三方网站,只要获取了唯一标识,基本上能抓取一些信息. 虾米 音乐的I ...

  2. K8s的POD连接数据库时报错

    [root@cccc xxxx]# ./showlog.sh dr iff-dr-1128668949-lb90g 2017-09-29 03:21:57,575 INFO [org.wildfly. ...

  3. 【JS新手教程】浏览器弹出div层1

    JS中,可以弹出一个层来进行提示等作用,方法是利用css样式display样式,当display等于none时,该元素就不会在页面显示出来,而且元素也不会占空间.就是用户触发某些事件时,动态修改该样式 ...

  4. 大数据-HBase HA集群搭建

    1.下载对应版本的Hbase,在我们搭建的集群环境中选用的是hbase-1.4.6 将下载完成的hbase压缩包放到对应的目录下,此处我们的目录为/opt/workspace/ 2.对已经有的压缩包进 ...

  5. springcloud(七)-Feign声明式REST调用

    前言 前面我们使用的RestTemplate实现REST API调用,代码大致如下: public User findById(@PathVariable Long id) { return rest ...

  6. 基于Cython和内置distutils库,实现python源码加密(非混淆模式)

    起因 python本身只能做混淆,不能加密,多年的商业软件开发经验导致有某种"洁癖"欲望,将py编译打包 尝试 pyinstaller原理是freeze打包pyc文件,利用工具可完 ...

  7. Q673 最长递增子序列的个数

    给定一个未排序的整数数组,找到最长递增子序列的个数. 示例 1: 输入: [1,3,5,4,7] 输出: 2 解释: 有两个最长递增子序列,分别是 [1, 3, 4, 7] 和[1, 3, 5, 7] ...

  8. 【Three.js】实现随心所欲的展示外部三维模型

    1.概要 最近学习Three.js,尝试加载一些3d max导出的obj.stl模型,在展示模型的时候遇到了一些问题,模型的尺寸.位置和旋转角度每次都靠手工调整,非常的不方便,就想着写一个方法来随心所 ...

  9. 在Windows Server 2008 R2(x64)上安装.NET Framework 4.5 兼谈.NET Framework 4.0 “在服务器核心角色上不受支持”含义

    完成了一个服务器文件监控系统,该系统的核心是一个Windows服务,需要安装在服务器上.由于是Visual Studio 2012开发,为了保证开发的Windows服务可以运行,必须在Windows服 ...

  10. 为apache提供sftp文件传输服务

    一.安装apache yum install httpd 二.为 /var/www/html 创建ftp账号www useradd -M -d /var/www/html www 三.更改ssh配置文 ...