CF427D
CF427D
SA的奇技淫巧,其实就是板子。
题意:
给定两个字符串,求最短的满足各只出现一次的连续公共字串
解析:
一般情况下,SA都是用来求最长公共前缀的,好像和这道题所求的最短公共子串没有任何关系。
但我们依然可以通过类比思路得出:想一想为什么要寻找zz最大的元素?
因为如果小于最大值,那么最大值就会包含这个序列。
所以答案就是,一个元素,没有z值比这个元素大的,自然就是要选z的最大值
从上述思路,寻找如何让答案尽量小
z的次大值自然是不行的,但是发现次大值+1是满足条件的
一方面,它比最大值小,所以被唯一的最大值包含;另一方面,它比次大值大,所以仅被最大值包含
所以可证,次大值+1也是唯一的。
所以,按如上方法,对 $ S_1 $ 的每一个后缀求最大值和次大值,再用次大值更新答案
然后我们就能愉快的用SA解决这个问题了。
CODE:
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
#define LL long long
#define N 10010
string s1,s2,str;
int SA[N],rk[N],tp[N],cnt[N];
int len,tot,m,a[N],height[N];
void qsort() {
for(int i = 1 ; i <= m ; i++) cnt[i] = 0;
for(int i = 1 ; i <= tot; i++) cnt[rk[i]]++;
for(int i = 1 ; i <= m ; i++) cnt[i] += cnt[i - 1];
for(int i = tot ; i >= 1 ; i--) SA[cnt[rk[tp[i]]]--] = tp[i];
}
inline bool cmp(int *f,int x,int y,int w) {
return f[x] == f[y] && f[x + w] == f[y + w];
}
void build_SA() {
m = 127;
for(int i = 1 ; i <= tot ; i++) {
rk[i] = a[i];
tp[i] = i;
}
qsort();
for(int w = 1 , p = 0 ; p < tot ; w += w,m = p) {
p = 0;
for(int i = tot - w + 1 ; i <= tot ; i++) tp[++p] = i;
for(int i = 1 ; i <= tot ; i++) {
if(SA[i] > w) tp[++p] = SA[i] - w;
}
qsort();
swap(rk,tp);
rk[SA[1]] = p = 1;
for(int i = 2 ; i <= tot ; i++)
rk[SA[i]] = cmp(tp,SA[i],SA[i - 1],w) ? p : ++p;
}
int j = 0, k = 0;
for(int i = 1 ; i <= tot ; height[rk[i++]] = k) {
for(k = k ? k - 1 : k, j = SA[rk[i] - 1] ; a[i + k] == a[j + k] ; k++);
}
}
inline bool check(int k,int div) {
int cnt1 = 0,cnt2 = 0;
for(int i = 1 ; i <= tot ; i++) {
if(height[i] < k) {
if(cnt1 == 1 && cnt2 == 1) return true;
cnt1 = cnt2 = 0;
if(SA[i] <= div) cnt1++;
else if(SA[i] >= div) cnt2++;
continue;
}
if(SA[i] <= div) cnt1++;
else if(SA[i] >= div) cnt2++;
}
return cnt1 == 1 && cnt2 == 1;
}
int main() {
cin>>s1>>s2;
len = s1.length();
str = s1 + '#' + s2;//加入'#'表示两个字符串的分界点。
tot = len + s2.length() + 1;
for(int i = 1 ; i <= tot ; i++) a[i] = str[i - 1];
build_SA();
int ans = -1;
for(int i = 1 ; i <= len ; i++) {
if(check(i,len)) {
ans = i;
break;
}
}
printf("%d \n",ans);
//system("pause");
return 0;
}
CF427D的更多相关文章
随机推荐
- NOIP2009-2018简要题解
口胡警告 NOIP2009 潜伏者 模拟 Hankson 的趣味题 对四个数\(a_0,a_1,b_0,b_1\)分解质因数,结果序列分别记为\(\{p1^{b1}\},\{p2^{b2}\},\{p ...
- oc和swift对代码的分组,方便代码查找和导航用
OC中对代码的分组: 在OC中对代码分组的命令是#pragma mark 来实现的, 分组用的是: #pragma mark 分组名 添加分割线: #pragma mark - 如果想添加分割线的同时 ...
- python现状
自从官方宣布 2020 年 1 月后不再更新维护 Python2,已经有一大批开源软件将其抛弃.今天,抛弃 Python2 的名单上又多了一个重磅软件.Python2 是 Python 官方在 200 ...
- echarts和v-chart使用心得
echarts的响应式 每次窗口大小改变的时候都会触发onresize事件,这个时候我们将echarts对象的尺寸赋值给窗口的大小这个属性,从而实现图表对象与窗口对象的尺寸一致的情况window.on ...
- # 机器学习算法总结-第六天(Adaboost算法)
SKlearn中的Adaboost使用 主要调的参数:第一部分是对我们的Adaboost的框架进行调参, 第二部分是对我们选择的弱分类器进行调参. 使用 Adaboost 进行手写数字识别 导入库,载 ...
- Java远程通讯可选技术及原理
转自:https://www.linuxidc.com/index.htm 在分布式服务框架中,一个最基础的问题就是远程服务是怎么通讯的,在Java领域中有很多可实现远程通讯的技术,例如:RMI.MI ...
- Asp.net Core 微信小程序支付
最近要做一个微信小程序支付的功能 在网上找了一下 .net Core做微信支付的博客 和 demo 几乎没有 自己研究了好几天 参考了 很多 大牛的博客 勉强做出来了 因为参数都没有 比如 opid ...
- boost交叉编译
运行bootstrap.sh # ./bootstrap.sh 生成b2.bjam和project-config.jam文件 修改project-config.jam using gcc : arm ...
- 【Distributed】CDN
一.概述 1.1 Web前端优化 1.2 DNS域名解析过程 1.3 传统方式请求静态资源 二.CDN内容分发 2.1 什么是CDN 2.2 CDN内容分发原理 2.3 阿里云环境实战搭建CDN内容分 ...
- web开发:javascript之dom与bom
一.节点认知 二.文档结构 三.文档节点操作 四.事件target 五.BOM操作 一.节点认知 - dom与dom属性 <!DOCTYPE html> <html> < ...