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 ...
随机推荐
- 机器学习笔记(五) K-近邻算法
K-近邻算法 (一)定义:如果一个样本在特征空间中的k个最相似的样本中的大多数属于某一个类别,则该样本也属于这个类别. (二)相似的样本,特征之间的值应该是相近的,使用k-近邻算法需要做标准化处理.否 ...
- Keil软件编程问题归总
Q:KEIL.uVision.RealView.MDK.KEIL C51之间的关系纠葛(比较区别)? A: KEIL是公司的名称,有时候也指KEIL公司的所有软件开发工具,目前2005年Keil由AR ...
- 2016级算法第六次上机-F.AlvinZH的学霸养成记VI
1082 AlvinZH的学霸养成记VI 思路 难题,凸包. 分析问题,平面上给出两类点,问能否用一条直线将二者分离. 首先应该联想到这是一个凸包问题,分别计算两类点的凸包,如果存在符合题意的直线,那 ...
- CODEVS-1215迷宫
迷宫 原题:传送门 题目描述 Description 在N*N的迷宫内,“#”为墙,“.”为路,“s”为起点,“e”为终点,一共4个方向可以走.从左上角((0,0)“s”)位置处走到右下角((n-1, ...
- 论文分享NO.2(by_xiaojian)
论文分享第二期-2019.03.26 NIPS2015,Spatial Transformer Networks,STN,空间变换网络
- PHP 导入数据库 sql 文件
使用PHP 可以导入sql来建立数据库.代码如下: <?php $hostname = 'localhost'; $dbname = 'test'; $username = 'root'; $p ...
- [问题解决]gradle编译失败系统找不到指定的文件
[问题解决]gradle编译失败系统找不到指定的文件 问题描述 Error:C:\Users\diql.gradle\caches\2.14.1\scripts-remapped\settings_9 ...
- Git学习系列之Git基本操作拉取项目(图文详解)
前面博客 Git学习系列之Git基本操作推送项目(图文详解) 当然,如果多人协作,或者多个客户端进行修改,那么我们还要拉取(Pull ... )别人推送到在线仓库的内容下来. 大神们是不推荐使用 pu ...
- JavaScript设计模式-3.原型模式
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- PHP之string之explode()函数使用
explode (PHP 4, PHP 5, PHP 7) explode - Split a string by string explode - 使用一个字符串分割另一个字符串 Descripti ...