【题目链接】:http://www.lydsy.com/JudgeOnline/problem.php?id=1014

【题意】



让你在线查询最长公共前缀.

支持单节点修改;

插入操作;

【题解】

/*
伸展树会保证
这棵树的中序遍历的结果是s[1..n]
即整个序列;
在进行旋转操作的时候,这个性质能被保持住;
伸展树在维护的时候;
每次会把需要操作的节点转到根节点;
然后进行对应的操作;
我们在进行
LCQ(x,y)的时候,
先二分枚举长度len;
然后把
x-1,x+len的节点编号获取a1,a2;
y-1,y+len的节点编号也获取b1,b2;
然后把a1转到根节点,a2放到根节点(也就是a1)的下面;
因为x+len>x-1所以a2肯定是在a1的右儿子处;
而这个时候a2的左子树代表的字符就是
s[x..x+len-1]了;
相应的对b1,b2也做同样的事情
也能获取s[y..y+len-1];
根据伸展树维护的hash值判断这两个子串是否相同。。
如果相同的话,就可以让len边长一点;
不同的话,肯定不能变长了,就变短一点呗.
然后返回答案就好 插入操作的话;
先提取x节点,把它转到根节点的位置;
然后再提取x+1号节点,把它转到根节点的下方
这里x+1号节点的左儿子肯定是空的,因为x和x+1是连在一起的;
中间不可能还有比x+1小的了;
则把这个新插入的节点放在x+1号节点的左边. 修改操作就简答多了
直接找到那个节点;
然后把它转到根节点去;
再修改它的值;
这里记住每次都把节点转到根节点就好 头部和尾部都要加一个空节点;
这样做写插入操作会好写一点吧? hash值可以搜一下RKhash;
这里不用管它会溢出;
你开一个unsigned long long就可以毁天灭地了;
let it go~~
*/

【完整代码】

#include <bits/stdc++.h>
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define LL unsigned long long
#define rep1(i,a,b) for (int i = a;i <= b;i++)
#define rep2(i,a,b) for (int i = a;i >= b;i--)
#define mp make_pair
#define pb push_back
#define fi first
#define se second
#define rei(x) scanf("%d",&x)
#define rel(x) scanf("%lld",&x)
#define which(x) (ch[fa[x]][1]==x) typedef pair<int, int> pii;
typedef pair<LL, LL> pll; const int dx[9] = { 0,1,-1,0,0,-1,-1,1,1 };
const int dy[9] = { 0,0,0,-1,1,-1,1,-1,1 };
const double pi = acos(-1.0);
const int N = 1e5 + 200;
const LL seed = 131; char s[N], str[N], op[10], val[10];
int m, n, fa[N], tot, ch[N][2], root, x, siz[N];
LL po_w[N], has[N]; void push_up(int x)
{
siz[x] = siz[ch[x][0]] + siz[ch[x][1]] + 1;
has[x] = has[ch[x][0]] + po_w[siz[ch[x][0]]] * str[x] + po_w[siz[ch[x][0]] + 1] * has[ch[x][1]];
} int build(int l, int r, int rt)
{
if (l > r) return 0;
int mid = (l + r) >> 1;
int x = ++tot;
fa[x] = rt; str[x] = s[mid];
ch[x][0] = build(l, mid - 1, x);
ch[x][1] = build(mid + 1, r, x);
push_up(x);
return x;
} int Rank(int x, int k)
{
if (siz[ch[x][0]] >= k)
return Rank(ch[x][0], k);
else
if (k == siz[ch[x][0]] + 1)
return x;
else
return Rank(ch[x][1], k - siz[ch[x][0]] - 1);
} void Rotate(int x)
{
int f = fa[x];
bool k = which(x);
ch[f][k] = ch[x][!k];
ch[x][!k] = f;
ch[fa[f]][which(f)] = x;
fa[ch[f][k]] = f;
fa[x] = fa[f];
fa[f] = x;
siz[x] = siz[f], has[x] = has[f];
push_up(f);
} void Splay(int x, int g)
{
while (fa[x] != g)
{
int f = fa[x];
if (fa[f] == g)
{
Rotate(x);
break;
}
if (which(x) ^ which(f))
Rotate(x);
else
Rotate(f);
Rotate(x);
}
if (!g) root = x;
} void Change(int pos, char val)
{
int x = Rank(root, pos);
Splay(x, 0);
str[x] = val;
push_up(x);
} void Insert(int pos, char val)
{
int x = Rank(root, pos), y = Rank(root, pos + 1);
Splay(x, 0), Splay(y, x);
ch[y][0] = ++tot;
str[tot] = val, fa[tot] = y;
push_up(tot), push_up(y), push_up(x);
} int lcq(int tx, int ty)
{
int l = 0, r = n, ans = 0;
while (l <= r)
{
int mid = (l + r) >> 1;
if (ty + mid - 1 > n + 1) //?????
{
r = mid - 1;
continue;
}
//[tx..tx+mid-1] but (tx-1,tx+mid)
int x = Rank(root, tx - 1), y = Rank(root, tx + mid);
Splay(x, 0), Splay(y, x);
LL temp1 = has[ch[y][0]];
x = Rank(root, ty - 1), y = Rank(root, ty + mid);
Splay(x, 0), Splay(y, x);
if (temp1 == has[ch[y][0]])
{
ans = mid;
l = mid + 1;
}
else
r = mid - 1;
}
return ans;
} int main()
{
//freopen("F:\\rush.txt", "r", stdin);
po_w[0] = 1;
rep1(i, 1, N - 2)
po_w[i] = po_w[i - 1] * seed;
scanf("%s", s + 1);
n = strlen(s + 1);
root = build(0, n + 1, 0);
rei(m);
rep1(i, 1, m)
{
scanf("%s", op);
if (op[0] == 'R')
{
scanf("%d%s", &x, val);
Change(x + 1, val[0]);
}
else
if (op[0] == 'I')
{
scanf("%d%s", &x, val);
Insert(x + 1, val[0]);
n++;
}
else
if (op[0] == 'Q')
{
int x, y;
scanf("%d%d", &x, &y);
if (x > y)
swap(x, y);
if (x != y)
printf("%d\n", lcq(x + 1, y + 1));
else
printf("%d\n", n - x + 1);
}
}
return 0;
}

【BZOJ 1014】 [JSOI2008]火星人prefix的更多相关文章

  1. BZOJ 1014: [JSOI2008]火星人prefix [splay 二分+hash] 【未完】

    1014: [JSOI2008]火星人prefix Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 6243  Solved: 2007[Submit] ...

  2. BZOJ 1014: [JSOI2008]火星人prefix Splay+二分

    1014: [JSOI2008]火星人prefix 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=1014 Description 火星人 ...

  3. bzoj 1014: [JSOI2008]火星人prefix hash && splay

    1014: [JSOI2008]火星人prefix Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3154  Solved: 948[Submit][ ...

  4. 求帮看!!!!BZOJ 1014 [JSOI2008]火星人prefix

    1014: [JSOI2008]火星人prefix Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4164  Solved: 1277[Submit] ...

  5. BZOJ 1014: [JSOI2008]火星人prefix( splay + hash )

    用splay维护序列, 二分+hash来判断LCQ.. #include<bits/stdc++.h> using namespace std; typedef unsigned long ...

  6. BZOJ 1014 [JSOI2008]火星人prefix (Splay + Hash + 二分)

    1014: [JSOI2008]火星人prefix Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 8112  Solved: 2569[Submit] ...

  7. [BZOJ 1014] [JSOI2008] 火星人prefix 【Splay + Hash】

    题目链接:BZOJ - 1014 题目分析 求两个串的 LCP ,一种常见的方法就是 二分+Hash,对于一个二分的长度 l,如果两个串的长度为 l 的前缀的Hash相等,就认为他们相等. 这里有修改 ...

  8. BZOJ 1014: [JSOI2008]火星人prefix

    Sol Splay+Hash+二分答案. 用Splay维护Hash,二分答案判断. 复杂度 \(O(nlog^2n)\) PS:这题调了两个晚上因为没开long long.许久不写数据结构题感觉写完整 ...

  9. bzoj 1014 [JSOI2008]火星人prefix(splay+hash)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1014 [题意] 给定一个字符串,要求提供修改一个字符,插入一个字符,查询两个后缀LCP ...

  10. bzoj 1014 [JSOI2008]火星人prefix——splay+哈希

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1014 用splay维护字符串,每个点记录子树的哈希值,然后二分查询. 二分不是把两个点的哈希 ...

随机推荐

  1. 学习笔记:Vue——插槽

    关于Vue插槽,只用过最简单的语法,现在完整地走一遍官方文档说明,并且探索更多用法. 01.如果组件中没有包含一个<slot>元素,则该组件起始标签和结束标签之间的任何内容都会被抛弃. 0 ...

  2. 【习题 6-11 UVA - 10410】Tree Reconstruction

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 可以先确定当前这棵子树的dfs序的范围. 然后第一个元素肯定是这棵子树的根节点. 那么只要在这棵子树的范围里面枚举节点. 看看有没有 ...

  3. 机器学习算法中怎样选取超參数:学习速率、正则项系数、minibatch size

    本文是<Neural networks and deep learning>概览 中第三章的一部分,讲机器学习算法中,怎样选取初始的超參数的值.(本文会不断补充) 学习速率(learnin ...

  4. (转)oracle表空间使用率统计查询

    转自:http://www.cnblogs.com/xwdreamer/p/3511047.html 参考文献 文献1:http://blog.itpub.net/24104518/viewspace ...

  5. 安装orabbix

    须知: (1). orabbix使用root用户安装. (2). orabbix安装在zabbix server端,而不是安装在Oracle端.   1.下载 Orabbix   2. 解压软件 un ...

  6. 【例题 6-5 UVA 12657 】Boxes in a Line

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 双向链表模拟题. 移动的时候,要注意它就在所需要的位置的情况.那种情况不移动. (如果已经在所需位置了,还用链表的插入方式强行移动的 ...

  7. RISC-V评估系列

    RISC-V评估系列 RISC-V工具链搭建 SiFive虚拟机分享--提取码:xe1c SiFive SDK函数结构 底层驱动 driver框架 操作系统FreeRTOS移植 FGPA评估 benc ...

  8. 【BZOJ 4518】[Sdoi2016]征途

    [链接] 链接 [题意] 在这里输入题意 [题解] DP+斜率优化; \(D(x) = E(x^2)-E(x)^2\) 其中\(E(x)^2\)这一部分是确定的. 因为总长是确定的,分成的段数又是确定 ...

  9. 【3005】拦截导弹问题(noip1999)

    Time Limit: 3 second Memory Limit: 2 MB 某国为了防御帝国的导弹袭击,开发出一种导弹拦截系统,但是这种拦截系统有一个缺陷:虽然他的第一发炮弹能达到任意的高度,但是 ...

  10. Html表单使用实例

    原文 https://www.jianshu.com/p/b01f32844ac1 大纲 1.单选框多选框实现的商品选择 2.添加下拉框和删除下拉框 3.观察textarea中事件处理器的运行顺序 推 ...