【JSOI2008】火星人prefix 哈希 非旋转treap
题目大意
就是给你一个字符串,有三种操作,共\(m\)个
\(Q~x~y\):询问第\(x\)个后缀和第\(y\)个后缀的LCP
\(R~x~y\):把第\(x\)个字符改成\(y\)
\(I~x~y\):在第\(x\)个字符后面插入一个字符\(y\)
\(m\leq 150000,\)任何时候字符串长度\(\leq 100000\),询问个数\(\leq 10000\)
题解
直接用平衡树维护哈希值,询问时二分答案。我用unsigned long long存回TLE,用unsigned存就能过。
时间复杂度:\(O(q\log^2n+m\log n)\)(\(q\)为询问个数)
代码
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<ctime>
#include<utility>
using namespace std;
typedef long long ll;
typedef unsigned ui;
typedef pair<int,int> pii;
struct node
{
int v;
int k;
int ls,rs;
int sz;
ui s;
node()
{
v=k=ls=rs=sz=0;
s=0;
}
};
node a[1000010];
int cnt=0;
ui pw[1000010];
int newnode(int v)
{
cnt++;
a[cnt].v=v;
a[cnt].k=rand();
a[cnt].s=v;
a[cnt].sz=1;
return cnt;
}
void mt(int p)
{
a[p].s=a[a[p].ls].s*pw[a[a[p].rs].sz+1]+a[p].v*pw[a[a[p].rs].sz]+a[a[p].rs].s;
a[p].sz=a[a[p].ls].sz+a[a[p].rs].sz+1;
}
int merge(int x,int y)
{
if(!x||!y)
return x+y;
if(a[x].k<a[y].k)
{
a[x].rs=merge(a[x].rs,y);
mt(x);
return x;
}
else
{
a[y].ls=merge(x,a[y].ls);
mt(y);
return y;
}
}
pii split(int x,int v)
{
if(!x)
return pii();
if(v<=a[a[x].ls].sz)
{
pii s=split(a[x].ls,v);
a[x].ls=s.second;
s.second=x;
mt(x);
return s;
}
else
{
pii s=split(a[x].rs,v-a[a[x].ls].sz-1);
a[x].rs=s.first;
s.first=x;
mt(x);
return s;
}
}
char s[1000010];
int rt=0;
int n;
ui gethash(int l,int r)
{
if(l>r)
return 0;
pii s1=split(rt,r);
pii s2=split(s1.first,l-1);
ui res=a[s2.second].s;
s1.first=merge(s2.first,s2.second);
rt=merge(s1.first,s1.second);
return res;
}
int query(int x,int y)
{
int l=0;
int r=min(n-x+1,n-y+1);
while(l<r)
{
int mid=(l+r+1)>>1;
if(gethash(x,x+mid-1)==gethash(y,y+mid-1))
l=mid;
else
r=mid-1;
}
return l;
}
void insert(int x,int v)
{
pii s1=split(rt,x);
s1.first=merge(s1.first,newnode(v));
rt=merge(s1.first,s1.second);
n++;
}
void change(int x,int v)
{
pii s1=split(rt,x);
pii s2=split(s1.first,x-1);
a[s2.second].s=a[s2.second].v=v;
s1.first=merge(s2.first,s2.second);
rt=merge(s1.first,s1.second);
}
int main()
{
freopen("bzoj1014.in","r",stdin);
freopen("bzoj1014.out","w",stdout);
srand(12700);
int i,m;
pw[0]=1;
for(i=1;i<=1000000;i++)
pw[i]=pw[i-1]*127;
scanf("%s",s+1);
n=strlen(s+1);
for(i=1;i<=n;i++)
rt=merge(rt,newnode(s[i]-'a'+1));
scanf("%d",&m);
int x,y;
for(i=1;i<=m;i++)
{
scanf("%s",s+1);
if(s[1]=='Q')
{
scanf("%d%d",&x,&y);
printf("%d\n",query(x,y));
}
else if(s[1]=='R')
{
scanf("%d%s",&x,s+1);
change(x,s[1]-'a'+1);
}
else
{
scanf("%d%s",&x,s+1);
insert(x,s[1]-'a'+1);
}
}
return 0;
}
【JSOI2008】火星人prefix 哈希 非旋转treap的更多相关文章
- BZOJ1014 JSOI2008 火星人prefix 【非旋转Treap】*
BZOJ1014 JSOI2008 火星人prefix Description 火星人最近研究了一种操作:求一个字串两个后缀的公共前缀.比方说,有这样一个字符串:madamimadam,我们将这个字符 ...
- 2018.06.28 BZOJ1014 [JSOI2008]火星人prefix(非旋treap+hash)
[JSOI2008]火星人prefix Time Limit: 10 Sec Memory Limit: 162 MB Submit: 8951 Solved: 2860 Description 火星 ...
- 【bzoj1014】: [JSOI2008]火星人prefix 平衡树-字符串-hash-二分
[bzoj1014]: [JSOI2008]火星人 用平衡树维护字符串的hash 然后询问的时候二分一下就好了 /* http://www.cnblogs.com/karl07/ */ #includ ...
- 1014: [JSOI2008]火星人prefix
1014: [JSOI2008]火星人prefix Time Limit: 10 Sec Memory Limit: 162 MB Description 火星人最近研究了一种操作:求一个字串两个后缀 ...
- 【bzoj1014】[JSOI2008]火星人prefix
1014: [JSOI2008]火星人prefix Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 6031 Solved: 1917[Submit] ...
- 【bzoj3224】Tyvj 1728 普通平衡树 01Trie姿势+平衡树的四种姿势 :splay,旋转Treap,非旋转Treap,替罪羊树
直接上代码 正所谓 人傻自带大常数 平衡树的几种姿势: AVL Red&Black_Tree 码量爆炸,不常用:SBT 出于各种原因,不常用. 常用: Treap 旋转 基于旋转操作和随机数 ...
- [BZOJ1014][JSOI2008]火星人prefix
[BZOJ1014][JSOI2008]火星人prefix 试题描述 火星人最近研究了一种操作:求一个字串两个后缀的公共前缀.比方说,有这样一个字符串:madamimadam,我们将这个字符串的各个字 ...
- BZOJ 1014: [JSOI2008]火星人prefix [splay 二分+hash] 【未完】
1014: [JSOI2008]火星人prefix Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 6243 Solved: 2007[Submit] ...
- BZOJ 1014: [JSOI2008]火星人prefix Splay+二分
1014: [JSOI2008]火星人prefix 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=1014 Description 火星人 ...
随机推荐
- 华为MAC Flapping , MAC的漂移
华为写的很详细,MAC 地址的漂移会导致流量的中断. 华为阻止MAC地址漂移的方法有三种: 一.端口配置静态MAC地址 在全局视图下,执行命令mac-address static mac-addres ...
- Mysql多实例之mysql服务脚本
1. #init port=3306 mysql_user="root" mysql_pwd="cancer" CmdPath="/applicati ...
- scrapy框架爬取妹子图片
首先,建立一个项目#可在github账户下载完整代码:https://github.com/connordb/scrapy-jiandan2 scrapy startproject jiandan2 ...
- Servlet 使用ServletContext共享数据,读取web.xml配置
ServletContext对象 session和cookie,对于每一个请求用户来说,都是不同的,因为要保证隐私安全. 而有一些数据,可以让所有用户共享,此时就可以用ServletContext对象 ...
- SQL Server中JOIN的使用方法总结
JOIN 分为:内连接(INNER JOIN).外连接(OUTER JOIN).其中,外连接分为:左外连接(LEFT OUTER JOIN).右外连接(RIGHT OUTER JOIN).全外连接(F ...
- java lang(Comparable接口) 和java util(Comparator接口)分析比较
//Comparable 接口强行对实现它的每个类的对象进行整体排序. -- 自然排序.类的compareTo称为自然比较方法. public interface Comparable<T> ...
- Html5使用canvas作图
以下例子是项目中实际用到的.不足之处请大家指正,设计到画线,写文字,填充,文字旋转. <!DOCTYPE html> <html> <head lang="en ...
- 428.x的n次幂
实现 pow(x,n) 不用担心精度,当答案和标准输出差绝对值小于1e-3时都算正确 样例 Pow(2.1, 3) = 9.261 Pow(0, 1) = 0 Pow(1, 0) = 1 挑战 O(l ...
- for循环游标
- C#中那些常用的工具类(Utility Class)(一)
代码越写越多,但是我们也需要经常去反思那些写过的代码,Utility Class就是这一类需要特别去反思总结的类,这些类像工具一样,我们经常通过一些静态方法,通过传入一些参数,然后得到我们需要的结果, ...