UVA11402 - Ahoy, Pirates!(线段树)

option=com_onlinejudge&Itemid=8&category=24&page=show_problem&problem=2397" target="_blank" style="">题目链接

题目大意:给你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!(线段树)的更多相关文章

  1. UVA 11402 - Ahoy, Pirates!(段树)

    UVA 11402 - Ahoy, Pirates! 题目链接 题意:总的来说意思就是给一个01串,然后有3种操作 1.把一个区间变成1 2.把一个区间变成0 3.把一个区间翻转(0变1,1变0) 思 ...

  2. 线段树(I tree)

    Codeforces Round #254 (Div. 2)E题这题说的是给了一个一段连续的区间每个区间有一种颜色然后一个彩笔从L画到R每个区间的颜色都发生了 改变然后 在L和R这部分区间里所用的颜色 ...

  3. bzoj3932--可持久化线段树

    题目大意: 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的 任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任务从第Si秒开始,在第 ...

  4. codevs 1082 线段树练习 3(区间维护)

    codevs 1082 线段树练习 3  时间限制: 3 s  空间限制: 128000 KB  题目等级 : 大师 Master 题目描述 Description 给你N个数,有两种操作: 1:给区 ...

  5. codevs 1576 最长上升子序列的线段树优化

    题目:codevs 1576 最长严格上升子序列 链接:http://codevs.cn/problem/1576/ 优化的地方是 1到i-1 中最大的 f[j]值,并且A[j]<A[i] .根 ...

  6. codevs 1080 线段树点修改

    先来介绍一下线段树. 线段树是一个把线段,或者说一个区间储存在二叉树中.如图所示的就是一棵线段树,它维护一个区间的和. 蓝色数字的是线段树的节点在数组中的位置,它表示的区间已经在图上标出,它的值就是这 ...

  7. codevs 1082 线段树区间求和

    codevs 1082 线段树练习3 链接:http://codevs.cn/problem/1082/ sumv是维护求和的线段树,addv是标记这歌节点所在区间还需要加上的值. 我的线段树写法在运 ...

  8. PYOJ 44. 【HNSDFZ2016 #6】可持久化线段树

    #44. [HNSDFZ2016 #6]可持久化线段树 统计 描述 提交 自定义测试 题目描述 现有一序列 AA.您需要写一棵可持久化线段树,以实现如下操作: A v p x:对于版本v的序列,给 A ...

  9. CF719E(线段树+矩阵快速幂)

    题意:给你一个数列a,a[i]表示斐波那契数列的下标为a[i],求区间对应斐波那契数列数字的和,还要求能够维护对区间内所有下标加d的操作 分析:线段树 线段树的每个节点表示(f[i],f[i-1])这 ...

随机推荐

  1. 使用 Eigen 3.3.3 进行矩阵运算

    Eigen是一个能够进行线性代数运算的C++开源软件包,包含矩阵和矢量操作,Matlab中对矩阵的大多数操作都可以在Eigen中找到. 最近需要计算厄米特矩阵的逆,基于LLT分解和LDLT分解,自己写 ...

  2. python爬取人民币汇率中间价

    python爬取人民币汇率中间价,从最权威的网站中国外汇交易中心. 首先找到相关网页,解析链接,这中间需要经验和耐心,在此不多说. 以人民币兑美元的汇率为例(CNY/USD),脚本详情如下: wind ...

  3. 磁盘镜像分析工具Autopsy

    磁盘镜像分析工具Autopsy   Autopsy是Kali Linux预安装的一款磁盘镜像分析工具.该工具可以对磁盘镜像的卷和文件系统进行分析,支持Unix和Windows系统.Autopsy是一个 ...

  4. hiho1291(逆序思维,并查集)

    题目链接:[https://hihocoder.com/problemset/problem/1291] 题意:在<我的世界>游戏中放置沙盒,沙盒为体积为1的正方体,按顺序给你一些坐标,然 ...

  5. QT学习笔记2:QT中常用函数

    一.QString转number QString number() QString number() QString number() QString number() QString number( ...

  6. 矩阵乘法<简单总结>

    原理:矩阵相乘最重要的方法是一般矩阵乘积.它只有在第一个矩阵的 行数 和第二个矩阵的 列数 相同时才可进行.若A为m×n矩阵,B为n×p矩阵,则他们的乘积AB会是一个m×p矩阵. 若A=    a   ...

  7. [HEOI2013]SAO

    题目大意: 一个有向无环图上有n个结点, 现在告诉你n-1个条件(x,y),表示x和y的先后关系. 问原图共有几种可能的拓扑序? 思路: 树形DP. f[i][j]表示对于第i个结点,有j个点在它前面 ...

  8. Qt on android 蓝牙开发(控制小车)

    因为要做一个用蓝牙控制小车的app,就用着QT搞了下,网上关于QT蓝牙开发的资料比较少,我在这里记录下过程希望对看到了人有所帮助 首先在项目文件里添加 QT += bluetooth 这样就可以用QT ...

  9. 一、 Log4E插件下载

    下载地址:http://log4e.jayefem.de/content/view/3/2/ 二.安装Log4E插件 将下载下来的压缩包解压缩,如下图所示: 解压缩生成的[de.jayefem.log ...

  10. 采集音频和摄像头视频并实时H264编码及AAC编码[转]

    0. 前言 我在前两篇文章中写了DirectShow捕获音视频然后生成avi,再进行264编码的方法.那种方法有一些局限性,不适合实时性质的应用,如:视频会议.视频聊天.视频监控等.本文所使用的技术, ...