UVA11402 - Ahoy, Pirates!(线段树)
UVA11402 - Ahoy, Pirates!(线段树)
题目大意:给你n个01串,每一个串拼接m次得到新串。最后在把这n个新串拼接起来得到终于的目标串。
然后给你四种操作:
F a b :把位置a到b都置为1;
E a b :把位置a到b都置为0。
I a b :把位置a到b上的数字翻转(0,1互换);
S a b :查询位置a到b有多少个1.
解题思路:线段树节点内部附加信息:setv:标记这个结点范围内有set值。而且须要传递到这个节点的孩子。resv:标记这个节点范围内有翻转。而且也须要传递到这个节点的孩子。
注意:setv和resv标记的范围是节点u的孩子,并不包含本身,所以还须要依据父亲节点附加信息处理这个节点的值。还有这题的时间非常紧。对于那些常常须要调用的函数加上inline能够加高速度。
代码:
#include <cstdio>
#include <cstring>
#define lson(x) (x<<1)
#define rson(x) ((x<<1) + 1)
const int maxn = 1100000;
int v[maxn];
struct Node {
int l, r, v;
int setv, resv;
void set (int l, int r, int v, int setv, int resv) {
this->l = l;
this->r = r;
this->v = v;
this->setv = setv;
this->resv = resv;
}
}node[4 * maxn];
inline void pushup (int u) {
node[u].set(node[lson(u)].l, node[rson(u)].r, node[lson(u)].v + node[rson(u)].v, -1, 0);
}
inline void set_node(int u, int v) {
node[u].setv = v;
node[u].resv = 0;
node[u].v = v * (node[u].r - node[u].l + 1);
}
inline void res_node(int u) {
node[u].resv ^= 1;
node[u].v = node[u].r - node[u].l + 1 - node[u].v;
}
inline void pushdown (int u) {
if (node[u].setv >= 0) {
set_node(lson(u), node[u].setv);
set_node(rson(u), node[u].setv);
node[u].setv = -1;
}
if (node[u].resv) {
res_node(lson(u));
res_node(rson(u));
node[u].resv = 0;
}
}
void build (int u, int l, int r) {
if (l == r)
node[u].set(l, r, v[l - 1], -1, 0);
else {
int m = (l + r) / 2;
build (lson(u), l, m);
build (rson(u), m + 1, r);
pushup(u);
}
}
void update (int u, int l, int r, int v) {
if (node[u].l >= l && node[u].r <= r) {
set_node(u, v);
return;
}
int m = (node[u].l + node[u].r) / 2;
pushdown(u);
if (l <= m)
update (lson(u), l, r, v);
if (r > m)
update (rson(u), l, r, v);
pushup(u);
}
void reserve (int u, int l, int r) {
if (node[u].l >= l && node[u].r <= r) {
res_node(u);
return;
}
int m = (node[u].l + node[u].r) / 2;
pushdown(u);
if (l <= m)
reserve(lson(u), l, r);
if (r > m)
reserve(rson(u), l, r);
pushup(u);
}
int query (int u, int l, int r) {
if (node[u].l >= l && node[u].r <= r)
return node[u].v;
int m = (node[u].l + node[u].r) / 2;
int ret = 0;
pushdown(u);
if (l <= m)
ret += query (lson(u), l, r);
if (r > m)
ret += query (rson(u), l, r);
pushup(u);
return ret;
}
int main () {
int T, n, t, q;
int x, y;
int N, len;
char s[100];
scanf ("%d", &T);
for (int i = 1; i <= T; i++) {
printf ("Case %d:\n", i);
memset (v, 0, sizeof (v));
len = 0;
scanf ("%d", &n);
while (n--) {
scanf ("%d%s", &t, s);
N = strlen (s);
for (int j = 0; j < t; j++)
for (int k = 0; k < N; k++) {
if (s[k] == '1')
v[len] = 1;
len++;
}
}
build(1, 1, len);
scanf ("%d", &q);
int cas = 0;
while (q--) {
scanf ("%s%d%d", s, &x, &y);
if (s[0] == 'F')
update(1, x + 1, y + 1, 1);
else if (s[0] == 'E')
update (1, x + 1, y + 1, 0);
else if (s[0] == 'I')
reserve(1, x + 1, y + 1);
else
printf ("Q%d: %d\n", ++cas, query(1, x + 1, y + 1));
}
}
return 0;
}
UVA11402 - Ahoy, Pirates!(线段树)的更多相关文章
- UVA 11402 - Ahoy, Pirates!(段树)
UVA 11402 - Ahoy, Pirates! 题目链接 题意:总的来说意思就是给一个01串,然后有3种操作 1.把一个区间变成1 2.把一个区间变成0 3.把一个区间翻转(0变1,1变0) 思 ...
- 线段树(I tree)
Codeforces Round #254 (Div. 2)E题这题说的是给了一个一段连续的区间每个区间有一种颜色然后一个彩笔从L画到R每个区间的颜色都发生了 改变然后 在L和R这部分区间里所用的颜色 ...
- bzoj3932--可持久化线段树
题目大意: 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的 任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任务从第Si秒开始,在第 ...
- codevs 1082 线段树练习 3(区间维护)
codevs 1082 线段树练习 3 时间限制: 3 s 空间限制: 128000 KB 题目等级 : 大师 Master 题目描述 Description 给你N个数,有两种操作: 1:给区 ...
- codevs 1576 最长上升子序列的线段树优化
题目:codevs 1576 最长严格上升子序列 链接:http://codevs.cn/problem/1576/ 优化的地方是 1到i-1 中最大的 f[j]值,并且A[j]<A[i] .根 ...
- codevs 1080 线段树点修改
先来介绍一下线段树. 线段树是一个把线段,或者说一个区间储存在二叉树中.如图所示的就是一棵线段树,它维护一个区间的和. 蓝色数字的是线段树的节点在数组中的位置,它表示的区间已经在图上标出,它的值就是这 ...
- codevs 1082 线段树区间求和
codevs 1082 线段树练习3 链接:http://codevs.cn/problem/1082/ sumv是维护求和的线段树,addv是标记这歌节点所在区间还需要加上的值. 我的线段树写法在运 ...
- PYOJ 44. 【HNSDFZ2016 #6】可持久化线段树
#44. [HNSDFZ2016 #6]可持久化线段树 统计 描述 提交 自定义测试 题目描述 现有一序列 AA.您需要写一棵可持久化线段树,以实现如下操作: A v p x:对于版本v的序列,给 A ...
- CF719E(线段树+矩阵快速幂)
题意:给你一个数列a,a[i]表示斐波那契数列的下标为a[i],求区间对应斐波那契数列数字的和,还要求能够维护对区间内所有下标加d的操作 分析:线段树 线段树的每个节点表示(f[i],f[i-1])这 ...
随机推荐
- mysql 之 用python操作数据库
- FastReport.Net使用:[14]文本控件使用
文本控件(Text)是FastReport中最常用的控件了,它可以是一行\多行文本.数据源的列.报表参数.汇总值.表达式,它还可以是以上任何元素的组合. 如何使用文本编辑器 1.双击文本框进入文本编辑 ...
- FastReport.Net使用:[13]如何使用表达式
在FastReport报表中,表达式(Expressions)用在很多地方,譬如文本框,排序过滤器等. 表达式基于报表选择的脚本语言,从菜单[报表]->[选项]打开“报表选项对话框”,切换到“脚 ...
- bzoj 3809 莫队
收获: 1.分块时顺便记录每个位置所属的块,然后一次排序就OK了. 2.要权衡在“区间移动”与“查询结果”之间的时间,莫队算法一般区间移动频率远大于查询结果,所以我们选择的辅助数据结构时就要注意了,我 ...
- 03-MyBatis主从实现代码读写分离应用以及实现
建立目录结构:
- HDU 5298 Solid Geometry Homework 暴力
Solid Geometry Homework 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5298 Description Yellowstar ...
- 【原】Eclipse部署Maven web项目到tomcat服务器时,没有将lib下的jar复制过去的解决办法
我们在做web开发是,经常都要在eclipse中搭建web服务器,并将开发中的web项目部署到web服务器进行调试,在此,我选择的是tomcat服务器.之前部署web项目到tomcat进行启动调试都很 ...
- 【转】Internet连接正常但是没有网络,禁用以太网以后再重新启动就可以使用了,原因是什么?
只是粘贴别人的答案,觉得有理,就放在博客里方便以后再学习~ 这个和网络中hdcp服务有关,网卡要在网路中通讯就必须要网络设备一般是路由器或者交换机分配地址,只有给了你电脑门牌号,信件投递能准确无误.你 ...
- EXTJS下拉树ComboBoxTree参数提交及回显方法
http://blog.csdn.net/wjlht/article/details/6085245 使用extjs可以构造出下拉数,但是不方便向form提交参数,在此,笔者想到一个办法,很方便Com ...
- [ext]form.submit()相关说明
form.submit({ url:"../addOrUpdatePack.shtml",method:'POST',success:function(f,action) { ...