cf213E 线段树维护hash
链接
https://codeforces.com/contest/213/problem/E
题目大意
给出两个排列a、b,长度分别为n、m,你需要计算有多少个x,使
得\(a_1 + x; a_2 + x; a_3 + x、、、 a_n + x\) 是b 的子序列(不连续的那种)。
思路
巧妙啊
暴力直接扫会T
我们构造一个c数组,使得c[b[i]]=i
这样x+1到x+1+n就是一段连续的区间了233
插回去看看他们相对大小是不是和a数组相同
因为不连续所以线段树维护hash值,线段树按照c数组的原路插回去
查询这一段是否hash值相同
؏؏☝ᖗ乛◡乛ᖘ☝؏؏
不知道双进制有没有用
update 19.3.14
写的稍微有点不清楚
因为a数组是个排列,所以[1,n]总的范围是1到n无重复
这样我们拿b数组开刀,c[b[i]]=i就是下标存值,值存下标。
这样你把他们放回去区间长度为n的下标,和a数组下标比较一下相对大小。
看看是否相同就可以判断是否是子串了。
然后[1,m],[2,m+1],[3,m+2]挨着扫就行了。
错误
报错:你数组开小啦
我:ull炸了吗?
代码
#include <iostream>
#include <cstdio>
#define ull unsigned long long
using namespace std;
const int N=2e5+7;
const int mod[2]={233,1000000009};
int read() {
int x=0,f=1;char s=getchar();
for(;s>'9'||s<'0';s=getchar()) if(s=='-') f=-1;
for(;s>='0'&&s<='9';s=getchar()) x=x*10+s-'0';
return x*f;
}
int n,m,pos[N],a[N],b[N];
ull my_pow[2][N];
namespace seg {
#define ls rt<<1
#define rs rt<<1|1
struct node {
int siz;
ull tot[2];
}e[N<<2];
int cnt;
void pushup(int rt) {
e[rt].siz=e[ls].siz+e[rs].siz;
e[rt].tot[0]=(e[ls].tot[0]*my_pow[0][e[rs].siz+1]+e[rs].tot[0]);
e[rt].tot[1]=(e[ls].tot[1]*my_pow[1][e[rs].siz+1]+e[rs].tot[1]);
}
void update(int l,int r,int id,int k,int rt) {
if(l==r) {
if(k==-1) e[rt].siz=e[rt].tot[0]=e[rt].tot[1]=0;
else e[rt].siz=1,e[rt].tot[0]=e[rt].tot[1]=k;
return;
}
int mid=(l+r)>>1;
if(id<=mid) update(l,mid,id,k,ls);
else update(mid+1,r,id,k,rs);
pushup(rt);
}
}
int main() {
n=read(),m=read();
my_pow[1][1]=my_pow[0][1]=1;
for(int i=2;i<=n+1;++i) {
my_pow[0][i]=my_pow[0][i-1]*mod[0];
my_pow[1][i]=my_pow[1][i-1]*mod[1];
}
for(int i=1;i<=n;++i) a[i]=read();
for(int i=1;i<=m;++i) b[i]=read();
ull hash[2]={0,0},tmp[2]={0,0};
for(int i=1;i<=n;++i) {
hash[0]=(hash[0]*mod[0]+a[i]);
hash[1]=(hash[1]*mod[1]+a[i]);
tmp[0]=(tmp[0]*mod[0]+1);
tmp[1]=(tmp[1]*mod[1]+1);
}
for(int i=1;i<=m;++i) pos[b[i]]=i;
int ans=0;
for(int i=1;i<=m;++i) {
if(i>n) seg::update(1,m,pos[i-n],-1,1);
seg::update(1,m,pos[i],i,1);
if(i>=n&&seg::e[1].tot[0]==(tmp[0]*(i-n)+hash[0])&&
seg::e[1].tot[1]==(tmp[1]*(i-n)+hash[1])) ans++;
}
cout<<ans<<"\n";
return 0;
}
cf213E 线段树维护hash的更多相关文章
- MemSQL Start[c]UP 2.0 - Round 1 F - Permutation 思维+线段树维护hash值
F - Permutation 思路:对于当前的值x, 只需要知道x + k, x - k这两个值是否出现在其左右两侧,又因为每个值只有一个, 所以可以转换成,x+k, x-k在到x所在位置的时候是否 ...
- BZOJ 2124: 等差子序列 线段树维护hash
2124: 等差子序列 Description 给一个1到N的排列{Ai},询问是否存在1<=p1=3),使得Ap1,Ap2,Ap3,…ApLen是一个等差序列. Input 输入的第一行包含一 ...
- bzoj 2124 等差子序列 (线段树维护hash)
2124: 等差子序列 Time Limit: 3 Sec Memory Limit: 259 MBSubmit: 1922 Solved: 714[Submit][Status][Discuss ...
- Codeforces Round #321 (Div. 2) E Kefa and Watch (线段树维护Hash)
E. Kefa and Watch time limit per test 1 second memory limit per test 256 megabytes input standard in ...
- BZOJ 2124 线段树维护hash值
思路: http://blog.csdn.net/wzq_QwQ/article/details/47152909 (代码也是抄的他的) 自己写得垃圾线段树怎么都过不了 隔了两个月 再写 再挂 又隔了 ...
- 线段树 + 字符串Hash - Codeforces 580E Kefa and Watch
Kefa and Watch Problem's Link Mean: 给你一个长度为n的字符串s,有两种操作: 1 L R C : 把s[l,r]全部变为c; 2 L R d : 询问s[l,r]是 ...
- cf580E. Kefa and Watch(线段树维护字符串hash)
题意 $n$个数的序列,$m + k$种操作 1.$l , r, k$把$l - r$赋值为$k$ 2.$l, r, d$询问$l - r$是否有长度为$d$的循环节 Sol 首先有个神仙结论:若询问 ...
- CF213E Two Permutations 线段树维护哈希值
当初竟然看成子串了$qwq$,不过老师的$ppt$也错了$qwq$ 由于子序列一定是的排列,所以考虑插入$1$到$m$到$n-m+1$到$n$; 如何判断呢?可以用哈希$qwq$: 我们用线段树维护哈 ...
- BZOJ.1036 [ZJOI2008]树的统计Count ( 点权树链剖分 线段树维护和与最值)
BZOJ.1036 [ZJOI2008]树的统计Count (树链剖分 线段树维护和与最值) 题意分析 (题目图片来自于 这里) 第一道树链剖分的题目,谈一下自己的理解. 树链剖分能解决的问题是,题目 ...
随机推荐
- python3 TypeError: a bytes-like object is required, not 'str'
在学习<Python web开发学习实录>时, 例11-1: # !/usr/bin/env python # coding=utf-8 import socket sock = sock ...
- uvm设计分析——tlm
tlm模块,用来在不同模块之间实现实时通信,主要基于两个定义在通信双方的port类来实现. 两个port之间,通过connect函数,来拿到双方的class指针,进而调用对方的function. 但是 ...
- [13]Windows 内核情景分析 --- 网络通信
典型的基于tcpip协议套接字方式的网络通信模块层次: 应用程序 socket api WS2_32.dll socket irp Afd.sys tdi irp Tcpip.sys 回调函数接口 各 ...
- Rpgmakermv(7) Chronus.js说明与简要翻译
插件地址:https://github.com/triacontane/RPGMakerMV/blob/master/Chronus.js 日语版 ゲーム内で時刻と天候の概念を表現できるプラグインです ...
- pyinstaller将python脚本生成exe
一.下载pyinstaller 二.生成exe 下载pyinstaller 1.在C:\python27\Scripts目录下打开cmd界面,执行命令:pip install PyInstaller ...
- javascript(一):javascript基本介绍及基本语法
什么是javascript? javascript是一种直译型脚本语言,是一种动态类型.弱类型.基于原型的语言.(所谓“脚本语言”:指的是它不具有开发操作系统的能力,只是用来编写大型运用程序的脚本!) ...
- Marlin 溫度感應器 數值轉換對應表
Marlin 溫度感應器 數值轉換對應表 (2014/03/27)Update: 自己實測了這個自動產生的對應表,結果測得的溫度與實際值仍有相當大的誤差.看來還是要回頭用測量的方式來校正溫度... ...
- python 解码json数据并在一个OrderdDict中保留其顺序
一般来讲,JSON 解码会根据提供的数据创建dicts 或lists.如果你想要创建其他类型的对象,可以给json.loads() 传递object_pairs_hook 或object_hook 参 ...
- 大数据是什么?它和Hadoop又有什么联系?
随着近几年计算机技术和互联网的发展,“大数据”这个名词越来越多进入我们的视野.大数据的快速发展也在无时无刻影响着我们的生活. 那大数据究竟是什么呢? 首先,看看专家是怎么解释大数据的: 大数据就是多, ...
- Linux基础命令---间歇执行命令watch
watch watch指令可以间歇性的执行程序,将输出结果以全屏的方式显示,默认是2s执行一次.watch将一直运行,直到被中断. 此命令的适用范围:RedHat.RHEL.Ubuntu.CentOS ...