【BZOJ 3682】Phorni
题目描述
Phorni 是一个音之妖精,喜欢在你的打字机上跳舞。
一天,阳光映射到刚刚淋浴过小雨的城市上时,Phorni 用魔法分裂出了许多个幻影,从 1 到 n 编号。
她的每一个幻影都站在打出的字符串的一个位置上,多个幻影可以站在同一个位置上。
每一个幻影代表的字符串即为从它站立位置开始的后缀,注意站立位置是从右往左数的。
让我们形式化地描述一下,若第 i 个幻影站在 Pi 上,那么它所代表的字符串就是 S[L-Pi+1…L],其中 L 是字符串 S 的长度。
每一次,她会选一段编号区间 [l..r],而编号在这个区间中的幻影中,字典序最小的一个将跳一支舞,若有多个幻影字典序相同,选编号最小的。
当然由于 Phorni 还会在打字机上跳动,所以有时字符串的前面会加入一个字符。
当然这个打字机是带加密功能的。
字典序的比较:
将两个字符串逐位比较,长度不足的向后补 0 ( 0 小于任何字符) 。直到比出大小或判定相等。
比如 “pho” > “ph” , “pb” > “pab” 。
下标从 1 开始,保证涉及到的所有字符都为小写字母。
Sol
后缀平衡树板子题。
用平衡树动态维护 SA 数组,采用赋权值的方式达到快速比较两个点的先后关系。
查询就用线段树维护并查询就行了。
code:
#include<bits/stdc++.h>
using namespace std;
#define Set(a,b) memset(a,b,sizeof(a))
template<class T>inline void init(T&x){
x=0;char ch=getchar();bool t=0;
for(;ch>'9'||ch<'0';ch=getchar()) if(ch=='-') t=1;
for(;ch>='0'&&ch<='9';ch=getchar()) x=(x<<1)+(x<<3)+(ch-48);
if(t) x=-x;return;
}typedef long long ll;
typedef double db;
const int N=5e5+10;
int n,m,len,type;
char S[N];
int ls[N],rs[N],size[N],pos[N];
db val[N];
#define ls (u<<1)
#define rs (u<<1|1)
int Min[N<<2];
inline void update(int u){
Min[u]=Min[ls];
if(Min[rs]&&val[pos[Min[u]]]>val[pos[Min[rs]]]) Min[u]=Min[rs];
return;
}
inline void Update(int u,int l,int r,int p){
if(l==r) return void(Min[u]=p);
int mid=(l+r)>>1;
if(mid>=p) Update(ls,l,mid,p);
else Update(rs,mid+1,r,p);
return update(u);
}
void Build(int u,int l,int r){
if(l==r) return void(Min[u]=l);
int mid=(l+r)>>1;
Build(ls,l,mid),Build(rs,mid+1,r);
update(u);
}
int Query(int u,int l,int r,int L,int R){
if(l>=L&&r<=R) return Min[u];
int mid=(l+r)>>1;
if(mid>=R) return Query(ls,l,mid,L,R);
if(mid< L) return Query(rs,mid+1,r,L,R);
int LP=Query(ls,l,mid,L,mid);
int RP=Query(rs,mid+1,r,mid+1,R);
if(val[pos[LP]]<=val[pos[RP]]) return LP;
return RP;
}
#undef ls
#undef rs
int rt;
inline bool cmp(int i,int j){if(S[i]!=S[j]) return S[i]<S[j];return val[i-1]<val[j-1];}
int stk[N],top;
void dfs1(int u){if(!u)return;dfs1(ls[u]);stk[++top]=u;dfs1(rs[u]);return;}
void dfs2(int&u,db l,db r,int L,int R){
if(l>r) return;int MID=(L+R)>>1;
u=stk[MID];size[u]=1;ls[u]=rs[u]=0;
db mid=val[u]=(l+r)/2.00;
dfs2(ls[u],l,mid,L,MID-1);
dfs2(rs[u],mid,r,MID+1,R);
size[u]+=size[ls[u]]+size[rs[u]];
return;
}
void Rebuild(int&u,db l,db r) {top=0;dfs1(u);dfs2(u,l,r,1,top);}
const db alpha=0.75;
inline bool Check(int u){
const db LIM=size[u]*alpha;
if(!LIM) return 0;
if(LIM<=size[ls[u]]||LIM<=size[rs[u]]) return 1;
return 0;
}
void Insert(int&u,db l,db r,int p,bool reb){
if(!u) {u=p;val[u]=(l+r)/2.0;return;}
bool re=!Check(u);
if(cmp(u,p)) Insert(rs[u],(l+r)/2.00,r,p,reb&re);
else Insert(ls[u],l,(l+r)/2.00,p,reb&re);
if((!re)&&reb) Rebuild(u,l,r);return;
}
int main()
{
init(n),init(m),init(len),init(type);
scanf("%s",S+1);rt=0;reverse(S+1,S+1+len);
for(int i=1;i<=len;++i) Insert(rt,0,1,i,1);
for(int i=1;i<=n;++i) init(pos[i]);Build(1,1,n);
int last=0;
for(int i=1;i<=m;++i) {
char sss[10];scanf("%s",sss);
if(sss[0]=='I') {
int c;init(c);
if(type==1) c^=last;
S[++len]=c+'a';
Insert(rt,0,1,len,1);
}else if(sss[0]=='C'){
int x,p;init(x),init(p);
pos[x]=p;Update(1,1,n,x);
}else{
int l,r;init(l),init(r);
last=Query(1,1,n,l,r);
printf("%d\n",last);
}
}
return 0;
}
【BZOJ 3682】Phorni的更多相关文章
- 【BZOJ 1150】 1150: [CTSC2007]数据备份Backup (贪心+优先队列+双向链表)
1150: [CTSC2007]数据备份Backup Description 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味 的,因此你想设 ...
- Kruskal算法及其类似原理的应用——【BZOJ 3654】tree&&【BZOJ 3624】[Apio2008]免费道路
首先让我们来介绍Krukal算法,他是一种用来求解最小生成树问题的算法,首先把边按边权排序,然后贪心得从最小开始往大里取,只要那个边的两端点暂时还没有在一个联通块里,我们就把他相连,只要这个图里存在最 ...
- 【BZOJ 2957】楼房重建&&Codechef COT5 Count on a Treap&&【NOIP模拟赛】Weed 线段树的分治维护
线段树是一种作用于静态区间上的数据结构,可以高效查询连续区间和单点,类似于一种静态的分治.他最迷人的地方在于“lazy标记”,对于lazy标记一般随我们从父区间进入子区间而下传,最终给到叶子节点,但还 ...
- LCA 【bzoj 4281】 [ONTAK2015]Związek Harcerstwa Bajtockiego
[bzoj 4281] [ONTAK2015]Związek Harcerstwa Bajtockiego Description 给定一棵有n个点的无根树,相邻的点之间的距离为1,一开始你位于m点. ...
- 【BZOJ 1191】 [Apio2010]特别行动队 (斜率优化)
dsy1911: [Apio2010]特别行动队 [题目描述] 有n个数,分成连续的若干段,每段的分数为a*x^2+b*x+c(a,b,c是给出的常数),其中x为该段的各个数的和.求如何分才能使得各个 ...
- 【BZOJ 1096】 [ZJOI2007]仓库建设 (斜率优化)
1096: [ZJOI2007]仓库建设 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 3940 Solved: 1736 Description ...
- 【BZOJ 2132】圈地计划 && 【7.22Test】计划
两种版本的题面 Description 最近房地产商GDOI(Group of Dumbbells Or Idiots)从NOI(Nuts Old Idiots)手中得到了一块开发土地.据了解,这块土 ...
- -【线性基】【BZOJ 2460】【BZOJ 2115】【HDU 3949】
[把三道我做过的线性基题目放在一起总结一下,代码都挺简单,主要就是贪心思想和异或的高斯消元] [然后把网上的讲解归纳一下] 1.线性基: 若干数的线性基是一组数a1,a2,a3...an,其中ax的最 ...
- 【BZOJ 1032】 [JSOI2007]祖码Zuma
[题目链接]:http://www.lydsy.com/JudgeOnline/problem.php?id=1032 [题意] [题解] /* 设f[i][j]表示从第i个珠子开始的j个珠子被消除; ...
随机推荐
- MySQL 查看约束,添加约束,删除约束 添加列,修改列,删除列
查看表的字段信息:desc 表名; 查看表的所有信息:show create table 表名; 添加主键约束:alter table 表名 add constraint 主键 (形如:PK_表名) ...
- MariaDB(第三章)select
基本查询 ``` --查询基本使用(条件,排序,聚合函数,分组,分页) --创建学生表 create table students ( id int unsigned not null auto_in ...
- POJ3734 Block母函数入门
一段长度为n的序列,你有红黄蓝绿四种颜色的砖块,一块砖长度为1,问你铺砖的方案数,其中红黄颜色之和必须为偶数. #include <queue> #include <stack> ...
- python笔记之列表
python中列表使用list类. 创建一个列表:list1 = [1,2,3,4]使用逗号隔开每个元素,使用方括号包含起来,创建空列表直接使用list2 = [] #!/usr/bin/env py ...
- SpringBoot 使用Mybatis操作mysql示例
1.准备数据库 创建数据库 create databases baodanjia; 创建帐号 create user 'baodanjia'@'%' identified by '123456' gr ...
- Linux 磁盘卷扩容
首先识别磁盘,成功之后会显示在/dev下 [root@oracle01 ~]# fdisk /dev/sda ## /dev/sda为通过fdisk -l 查看到的物理磁盘(第一行) Welcome ...
- Linux 概念与快捷方式
概念 何为shell Shell 是指"提供给使用者使用界面"的软件(命令解析器),类似于 DOS 下的 command(命令行)和后来的 cmd.exe .普通意义上的 Shel ...
- [Python3] 035 函数式编程 高阶函数
目录 函数式编程 之 高阶函数 1. 引子 2. 系统提供的高阶函数 3. functools 包提供的 reduce 4. 排序 函数式编程 之 高阶函数 把函数作为参数使用的函数,叫高阶函数 1. ...
- PTA(Basic Level)1087.有多少不同的值
当自然数 n 依次取 1.2.3.--.N 时,算式 ⌊n/2⌋+⌊n/3⌋+⌊n/5⌋ 有多少个不同的值?(注:⌊x⌋ 为取整函数,表示不超过 x 的最大自然数,即 x 的整数部分.) 输入格式: ...
- [转帖]Java 8新特性探究 前言
Java 8新特性探究 前言 https://my.oschina.net/benhaile/blog/174136 讲下java的历史 感觉挺好的. 评论 17 jdk8java8javase新特性 ...