当前位置:网站首页>“蔚来杯“2022牛客暑期多校训练营3 H.Hacker SAM+线段树/DP/分治(不带修查区间最大子段和)
“蔚来杯“2022牛客暑期多校训练营3 H.Hacker SAM+线段树/DP/分治(不带修查区间最大子段和)
2022-07-28 15:27:00 【HeartFireY】
H.Hacker
题目分析
给定母串以及若干子串,子串长度固定,每个位置都有一个权值,要求在子串和母串的公共子串中找到一个连续区间,使得连续区间的权值和最大,求最大权值和。
首先对母串建立SAM,然后用线段树维护静态区间和的最大值,然后每次对插入的字符串:
我们设置两个变量 v , l v,l v,l,分别表示当前状态以及匹配长度,一开始 v = t 0 v=t_0 v=t0 且 l = 0 l=0 l=0,即匹配为空串。
现在我们来描述如何添加一个字符 T i T_{i} Ti 并为其重新计算答案:
- 如果存在一个从 v v v 到字符 T i T_{i} Ti 的转移,我们只需要转移并让 l l l 自增一。
- 如果不存在这样的转移,我们需要缩短当前匹配的部分,这意味着我们需要按照后缀链接进行转移:
v = link ( v ) v=\operatorname{link}(v) v=link(v)
与此同时,需要缩短当前长度。显然我们需要将 l l l 赋值为 len ( v ) \operatorname{len}(v) len(v)。
每次找到合法区间,直接线段树上查询并更新答案即可。
数据很水,不缩也能过,这里提供一组群友造的特殊样例:
输入:
11 11 1
aaaabbbaaaa
2 3 7 5 10 -2 0 10 2 4 0
aaaaaaaaaaa
输出:
25
Code
#include <bits/stdc++.h>
#pragma gcc optimize("O2")
#pragma g++ optimize("O2")
#define int long long
#define endl '\n'
using namespace std;
const int N = 2e5 + 10, MOD = 1e9 + 7;
const int INF = 1e9 + 7;
int ww[N];
#define ls rt << 1
#define rs rt << 1 | 1
#define lson rt << 1, l, mid
#define rson rt << 1 | 1, mid + 1, r
struct node{
int w, lw, rw, maxw;
node(){
w = 0;
lw = rw = maxw = -INF;
}
node operator+ (node b) const{
node c;
c.w = w + b.w;
c.lw = max(lw, w + b.lw);
c.rw = max(b.rw, b.w + rw);
c.maxw = max(max(maxw, b.maxw), rw + b.lw);
return c;
}
}tree[N << 2];
void build(int rt, int l, int r){
if(l == r){
tree[rt].w = tree[rt].lw = tree[rt].rw = tree[rt].maxw = ww[l];
return;
}
int mid = l + r >> 1;
build(lson), build(rson);
tree[rt] = tree[ls] + tree[rs];
}
node query(int rt, int l, int r, int L, int R){
if(l >= L && r <= R) return tree[rt];
int mid = l + r >> 1;
node b, c;
if(mid >= L) b = query(lson, L, R);
if(mid < R) c = query(rson, L, R);
return b + c;
}
struct state {
int len, link;
std::map<char, int> next;
};
const int MAXLEN = 100000;
state st[MAXLEN << 1];
int sz, last;
void sam_init() {
st[0].len = 0;
st[0].link = -1;
sz++;
last = 0;
}
void sam_extend(char c) {
int cur = sz++;
st[cur].len = st[last].len + 1;
int p = last;
while (p != -1 && !st[p].next.count(c)) {
st[p].next[c] = cur;
p = st[p].link;
}
if (p == -1) {
st[cur].link = 0;
} else {
int q = st[p].next[c];
if (st[p].len + 1 == st[q].len) {
st[cur].link = q;
} else {
int clone = sz++;
st[clone].len = st[p].len + 1;
st[clone].next = st[q].next;
st[clone].link = st[q].link;
while (p != -1 && st[p].next[c] == q) {
st[p].next[c] = clone;
p = st[p].link;
}
st[q].link = st[cur].link = clone;
}
}
last = cur;
}
int ql, qr;
int n, m, k;
int get(const string &T) {
int v = 0, l = 0, best = 0, bestpos = 0, ans = 0;
for (int i = 0; i < T.size(); i++) {
while (v && !st[v].next.count(T[i])) {
v = st[v].link;
l = st[v].len;
}
if (st[v].next.count(T[i])) {
v = st[v].next[T[i]];
l++;
}
int nowl = i - l + 1, nowr = i;
if(nowl <= nowr){
int res = query(1, 1, m, nowl + 1, nowr + 1).maxw;
ans = max(ans, res);
}
}
return ans;
}
inline void solve(){
cin >> n >> m >> k;
string a; cin >> a;
sam_init();
for (int i = 0; i < a.size(); i++) sam_extend(a[i]);
for(int i = 1; i <= m; i++) cin >> ww[i];
build(1, 1, m);
while(k--){
string s; cin >> s;
cout << get(s) << endl;
}
}
signed main(){
ios_base::sync_with_stdio(false), cin.tie(0);
cout << fixed << setprecision(12);
int t = 1; //cin >> t;
while(t--) solve();
return 0;
}
边栏推荐
- 视频号找到金钥匙,抖音模仿后来人
- Writing of factorial
- Remote serial port server (adapter) UART to 1-wire application
- Headline article_ signature
- Qt学习第一天
- 为什么学编程的人大多数都去了深圳和北京?
- KubeEdge发布云原生边缘计算威胁模型及安全防护技术白皮书
- 加速投资的小红书,“病急乱投医”?
- Use py to automatically generate weekly reports based on log records
- Image semantic segmentation practice: tensorflow deeplobv3+ train your own dataset
猜你喜欢

JS array (summary)

JS queue

KubeEdge发布云原生边缘计算威胁模型及安全防护技术白皮书

Dynamic programming -- digital statistics DP

Remote serial port server (adapter) UART to 1-wire application

5 亿用户,比微信还早四年……这个运营了 15 年的 APP 即将永久停服

MySQL view event status statements and modification methods

About the web docking pin printer, lodop uses

Food safety | these two kinds of melons and fruits should be improved, especially for pregnant women with constipation

RF module wireless transceiver rf63u chip application data transmission and infrastructure network
随机推荐
使用py,根据日志记录自动生成周报
ANSA二次开发 - 在PyCharm上搭建ANSA/META二次开发环境
R language ggplot2 visually draws line plots, and uses gghighlight package to highlight the lines that meet the combination judgment conditions in the line graphs (satisfies both condition a and b)
Brief tutorial for soft exam system architecture designer | software debugging
Advantages of optical rain gauge over tipping bucket rain gauge
PHP获取小程序码,小程序带参数跳转
R语言ggplot2可视化绘制线图(line plot)、使用gghighlight包突出高亮线图中满足组合判断条件的线图(satisfies both condition A and B)
自动打包压缩备份下载及删除 bat脚本命令
关于web对接针式打印机问题,Lodop使用
软考 系统架构设计师 简明教程 | 软件调试
High precision absolute angle sensor application high speed angle monitoring
R language uses ggpairs function of ggally package to visualize pairwise relationship graph of grouping multivariable, set alpha parameter to change image transparency, diagonal continuous variable de
JS linked list 02
Redis系列4:高可用之Sentinel(哨兵模式)
Laser rangefinder non-contact surface crack monitor
PHP 图片上传
Temperature measurement and imaging accuracy of ifd-x micro infrared imager (module)
mysql查询 limit 1000,10 和limit 10 速度一样快吗?如果我要分页,我该怎么办?
Zhengda cup hacker marathon data analysis competition
带你来浅聊一下!单商户功能模块汇总