UOJ#172. 【WC2016】论战捆竹竿
传送门
首先这个题目显然就是先求出所有的 \(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】论战捆竹竿的更多相关文章
- bzoj4406: [Wc2016]论战捆竹竿&&uoj#172. 【WC2016】论战捆竹竿
第二次在bzoj跑进前十竟然是因为在UOJ卡常致死 首先这个题其实就是一个无限背包 一般做法是同余最短路,就是bzoj2118: 墨墨的等式可以拿到30分的好成绩 背包是个卷积就分治FFT优化那么下面 ...
- luogu P4156 [WC2016]论战捆竹竿
传送门 官方题解(证明都在这) 神仙题鸭qwq 转化模型,发现这题本质就是一个集合,每次可以加上集合里的数,问可以拼出多少不同的数 首先暴力需要膜意义下的最短路,例题戳这 然后这个暴力可以优化成N^2 ...
- Luogu4156 WC2016 论战捆竹竿 KMP、同余类最短路、背包、单调队列
传送门 豪华升级版同余类最短路-- 官方题解 主要写几个小trick: \(1.O(nm)\)实现同余类最短路: 设某一条边长度为\(x\),那么我们选择一个点,在同余类上不断跳\(x\),可以形成一 ...
- BZOJ4406 WC2016 论战捆竹竿
Problem BZOJ Solution 显然是一个同余系最短路问题,转移方案就是所有|S|-border的长度,有 \(O(n)\) 种,暴力跑dijkstra的复杂度为 \(O(n^2\log ...
- 「WC2016」论战捆竹竿
「WC2016」论战捆竹竿 前置知识 参考资料:<论战捆竹竿解题报告-王鉴浩>,<字符串算法选讲-金策>. Border&Period 若前缀 \(pre(s,x)\ ...
- UOJ#172. 【WC2016】论战捆竹竿 字符串 KMP 动态规划 单调队列 背包
原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ172.html 题解 首先,这个问题显然是个背包问题. 然后,可以证明:一个字符串的 border 长度可 ...
- 【WC2016】论战捆竹竿
已经快三周了啊--终于把挖的坑填了-- 首先显然是把除了自身的所有border拿出来,即做 \(\left\{ n - b_1, n - b_2, \dots, n - b_k, n \right\} ...
- 【LuoguP4156】论战捆竹竿
题目链接 题意简述 你有一个长度为 n 的字符串 , 将它复制任意次 , 复制出的串的前缀可以与之前的串的后缀重叠在一起 , 问最后总共可能的长度数目 , 长度不能超过 \(w\) 多组数据. \(n ...
- bzoj AC倒序
Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...
随机推荐
- html中object和embed标签的区别
♦object定义一个嵌入的对象.请使用此元素向您的 XHTML 页面添加多媒体.此元素允许您规定插入 HTML 文档中的对象的数据和参数,以及可用来显示和操作数据的代码. ♦<object&g ...
- Python笔记之字典循环
Python笔记之字典循环 1.问题 Python是一门比较好入门的编程语言,但是入门简单,当然坑也是有的,今天就来介绍一个我遇到的坑吧,也是很简单的一个,就是当时脑子有点转不过弯来了. 先看代码 ...
- java开发注解大全
目录 1.最基础注解(spring-context包下的org.springframework.stereotype) 1.1.@Controller @Service @Repository @Co ...
- 收集vcftools所有用法
VCFtools用来处理VCF文档. 筛选特定突变 比较文件 总结突变 转化文件格式 验证并合并文件 取突变交集和差集 Get basic file statistics input可以为VCF或BC ...
- Mac下Homebrew的图形化界面工具Cakebrew
安装: brew cask install cakebrew 如果不能下载直接上官网下载dmg包进行安装. 参考: https://www.cakebrew.com/ https://github.c ...
- (转)史上最全的MSSQL复习笔记
1.什么是SQL语句 sql语言:结构化的查询语言.(Structured Query Language),是关系数据库管理系统的标准语言. 它是一种解释语言:写一句执行一句,不需要整体编译执行.语法 ...
- Linus' Law
Given enough eyeballs, all bugs are shallow. ------埃里克 ...
- python-Unix套接字
#!/usr/bin/python #coding=utf-8 #server import socket import sys import os server_address = './test' ...
- ubuntu工具安装
smplayer sudo add-apt-repository ppa:rvm/smplayer sudo apt-get update sudo apt-get install smplayer ...
- Flink初始
flink初始 flink是什么 为什么使用flink flink的基础概念 flink剖析 实例 flink是什么 flink是一个用于有界和无界数据流进行有状态的计算框架. flink提供了不同级 ...