CSUOJ 1542 线段树解决括号反向问题
题目大意:
根据初始给定的合法的小括号排序,每次进行一个操作,将第a位的括号反向,找到一个尽可能靠前的括号反向后是整个括号排列合法
数据量十分大,不断进行查询,要用线段树进行logn的复杂度的查询
首先最简单的考虑 '('->')' , 稍微想一下可以知道因为要尽可能靠前,所以其实把最前面的那个 )改成 ( 即可,这里我就用 minn[] 数组记录区间内最早出现的 ) 的下标
然后是考虑 ')'->'(' , 我们可以倒着字符串来看,从后往前每次出现一个 ) 都记录加1 , 那么每次遇到一个 ( 就抵消1 , 那么当遇到 (没东西抵消时,说明这个是离尾部最远的不合法的符号,离尾部最远,那么就可以理解为离起点最近
具体怎么写的话就是可以将 '(' 看作 1 , ')' 看作-1 , 利用一个数组 minsum[] 记录区间内前缀和的最小值
对于一个线段树来说如果那个符号的位置出现在左子树的区间上,那么右子树中的所有前缀和都必然 >=2
所以查询就很容易得到 if(sum[rs] < 2) ans = 右子树的查询,else ans = 左子树的查询
但再想想的话,返回的值不能是当前下标 , 而是下标+1;
因为根据前面所讲,你所需要修改的位置 i 是 i 后面的数相加和正好为0的
也就是说到达i的前缀之和正好是2
而 第 i-1 位的前缀和正好为1,每次不断判断找 < 2的点,那么最后查询到的是i-1
也就是必须要加个1才能到达我需要改的位置
#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std; #define ls o<<1
#define rs o<<1|1
#define define_m int m=(l+r)>>1
const int N = ;
const int INF = 0x3f3f3f3f;
int minn[N<<] , minsum[N<<] , a[N] , add[N<<];
char str[N]; void push_up(int o)
{
minsum[o]=min(minsum[ls] , minsum[rs]);
minn[o]=min(minn[ls] , minn[rs]);
} void push_down(int o)
{
if(add[o]){
add[ls]+=add[o];
add[rs]+=add[o];
minsum[ls]+=add[o];
minsum[rs]+=add[o];
add[o]=;
}
} void build(int o , int l , int r)
{
add[o]=;
if(l == r) {
minn[o] = str[l]==')'?l:INF;
minsum[o]=a[l];
return;
}
define_m;
build(ls , l , m);
build(rs , m+ , r);
push_up(o);
} void update(int o , int l , int r , int s , int t , int v)
{
if(l>=s && r<=t){
minsum[o]+=v;
add[o]+=v;
return;
}
push_down(o);
define_m;
if(m>=s) update(ls , l , m , s , t , v);
if(m<t) update(rs , m+ , r ,s , t , v);
minsum[o]=min(minsum[ls] , minsum[rs]);
} void update1(int o , int l , int r , int pos)
{
if(l == r && l == pos){
minn[o] = (str[l]==')'?l:INF);
return;
}
push_down(o);
define_m;
if(m >= pos) update1(ls , l , m ,pos);
else update1(rs , m+ , r , pos);
minn[o] = min(minn[ls] , minn[rs]);
} int query(int o , int l , int r , int n)
{
// cout<<"o: "<<o<<" l: "<<l<< " r: "<<r<<" left: "<<minsum[ls]<<" right: "<<minsum[rs]<<endl;
if(l==r) return l+;
push_down(o);
define_m;
if(minsum[rs]<) return query(rs , m+ , r , n);
else return query(ls , l , m , n);
} int main()
{
// freopen("a.in" , "r" , stdin);
int n , m , pos;
while(scanf("%d%d" , &n , &m) != EOF)
{
scanf("%s" , str+);
for(int i= ; i<=n ; i++){
a[i]=a[i-]+(str[i]=='('?:-);
}
build( , , n);
int res;
for(int i= ; i<m ; i++){
scanf("%d" , &pos);
if(str[pos] == '('){
str[pos] = ')';
update1( , , n , pos);
update( , , n , pos , n , -);
res = minn[];
str[res] = '(';
update1( , , n , res);
}
else{
str[pos] = '(';
update1( , , n , pos);
update( , , n , pos , n , );
res = query( , , n , n);
str[res] = ')';
update1( , , n , res);
}
update( , , n , res , n , str[res] == '('?:-);
printf("%d\n" , res);
}
}
return ;
}
CSUOJ 1542 线段树解决括号反向问题的更多相关文章
- hdu 1542 线段树扫描(面积)
Atlantis Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Su ...
- bzoj1095: [ZJOI2007]Hide 捉迷藏 线段树维护括号序列 点分治 链分治
这题真是十分难写啊 不管是点分治还是括号序列都有一堆细节.. 点分治:时空复杂度$O(n\log^2n)$,常数巨大 主要就是3个堆的初始状态 C堆:每个节点一个,为子树中的点到它父亲的距离的堆. B ...
- BZOJ 1095 捉迷藏(线段树维护括号序列)
对于树的一个括号序列,树上两点的距离就是在括号序列中两点之间的括号匹配完之后的括号数... 由此可以得出线段树的做法.. #include<cstdio> #include<iost ...
- [板子]用线段树解决ST表问题
ST表可以参考:http://blog.csdn.net/whistlena/article/details/52191463 简单说就是区间RMQ最值问题. 对解决这种问题,线段树不用用啥啊. 扔一 ...
- hdu 1542 线段树+扫描线 学习
学习扫描线ing... 玄学的东西... 扫描线其实就是用一条假想的线去扫描一堆矩形,借以求出他们的面积或周长(这一篇是面积,下一篇是周长) 扫描线求面积的主要思想就是对一个二维的矩形的某一维上建立一 ...
- hdu 1542 线段树之扫描线之面积并
点击打开链接 题意:给你n个矩形,求它们的面积,反复的不反复计算 思路:用线段树的扫描线完毕.将X坐标离散化后,从下到上扫描矩形,进行各种处理,看代码凝视把 #include <stdio.h& ...
- BZOJ 1095: [ZJOI2007]Hide 捉迷藏(线段树维护括号序列)
这个嘛= =链剖貌似可行,不过好像代码长度很长,懒得打(其实是自己太弱了QAQ)百度了一下才知道有一种高大上的叫括号序列的东西= = 岛娘真是太厉害了,先丢链接:http://www.shuizilo ...
- BZOJ1095: [ZJOI2007]Hide 捉迷藏【线段树维护括号序列】【思维好题】
Description 捉迷藏 Jiajia和Wind是一对恩爱的夫妻,并且他们有很多孩子.某天,Jiajia.Wind和孩子们决定在家里玩 捉迷藏游戏.他们的家很大且构造很奇特,由N个屋子和N-1条 ...
- CodeForces-380C:Sereja and Brackets(线段树与括号序列)
Sereja has a bracket sequence s1, s2, ..., sn, or, in other words, a string s of length n, consistin ...
随机推荐
- Linux环境下HDFS集群环境搭建关键步骤
Linux环境下HDFS集群环境搭建关键步骤记录. 介质版本:hadoop-2.7.3.tar.gz 节点数量:3节点. 一.下载安装介质 官网下载地址:http://hadoop.apache.or ...
- 214 Shortest Palindrome 最短回文串
给一个字符串 S, 你可以通过在字符串前面添加字符将其转换为回文串.找到并返回可以用这种方式转换的最短回文串.例如:给出 "aacecaaa",返回 "aaacecaaa ...
- HTML中的那些bug
1.语法检测时提示有多余的结束标签 <!doctype html> <html> <head> <meta charset="utf-8" ...
- iOS programming UITableView and UITableViewController
iOS programming UITableView and UITableViewController A UITableView displays a single column of dat ...
- testlink 从1.8.5 升级到 1.9.8
step1:备份原 1.8.5 的数据库. step2:分别下载 1.9.0 / 1.9.3 / 1.9.8 的安装包. step3:分别解压 1.9.0 / 1.9.3 / 1.9.8 成3 ...
- H1B工作签证·绿卡:美国留学的两个关键步骤
月20日在留美学生家长群聚会上的发言稿一.H1B签证系美国最主要的工作签证类别,发放给美国公司雇佣的外国籍有专业技能的员工,属于非移民签证的一种.持有H1B签证者可以在美国工作三年,然后可以再延长三年 ...
- 1、QQ装机部落---腾讯软件
亲,赚钱快人一步,我给你推荐QQ装机部落!大品牌,口碑好,知名软件,绿色无毒,每天结算,回报高!到这里注册:http://zjbl.qq.com/#/?invid=476058088 返利网邀请码立即 ...
- 博客之旅 gogogo!
听说写博客的人都很牛~ 上班一年多了,想记录点什么,so,就写博客吧,整理一些技术点与工作生活心得 欢迎各位道友交流学习 :)
- JVM:内存分配与回收策略
Java技术体系中所提倡的自动内存管理最终可以归结为自动化的解决了两个问题:给对象分配内存以及回收分配给对象的内存. 对象的内存分配,往大方向讲,就是在堆上分配(但也可能经过JIT编译后被拆散为标量类 ...
- 网络爬虫之框架(Scrapy)
Scrapy爬虫框架 爬虫框架是实现爬虫功能的一个软件结构和功能组件集合. 爬虫框架是一个半成品,能够帮助用户实现专业网络爬虫. Scrapy爬虫框架结构: