数据结构习题 线段树&树状数组
说明:这是去年写了一半的东西,一直存在草稿箱里,今天整理东西的时候才发现,还是把它发表出来吧。。
以下所有题目来自Lrj的《训练指南》
LA 2191
单点修改,区间和 Fenwick直接搞
UVa 12299
给出n个数,支持循环移动某些数(<30个),然后问区间最小值
因为移动小于30个数,所以直接单点修改就行,线段树。
LA 4108
类似线段树,每次插入一个建筑时想线段树一样二分区间,当遇到一个完整的区间时,修改或返回,否则继续二分区间。Nlogn.
UVa 11525
有题解说可以逆向考虑用线段树,我没明白怎么回事。
说一下我的做法:
考虑问题(k,n)表示1~k的全排列中第n个。显然1~k的全排列可以分为k种,每种有(k-1)!种情况,那么如果确定了第n个排列属于哪一种,问题就转化为了(k-1,m);
再考虑题目的输入。N的输入形式正好是按以上思想给出的,那么我们是需要把这个式子转化为标准形式即可。
判断每个Si,若Si>(k-i+1),就要”进位”。所以从后向前扫描,可以在O(n)时间内解决。
LA 4730
并查集+线段树。用并查集维护每个州,同时维护每个州的上端点(up)和下端点(down)。
用线段树维护[a,b]内有多少个城市。方法是连接A和B时,[A.down,A.up]-=A.count; [B.down,B.up]-=B.count; [new.down,new.up]+=new.count;
二维线段树
先存一下(以下代码仅供参考 准确性无法保证233)
#include <cstdio>
#include <cstdlib>
#include <iostream>
#include <cstring>
#define RS(x) (x*2+1)
#define LS(x) (x*2)
#define bigger(a,b) ((a)>(b)?(a):(b))
#define minner(a,b) ((a)<(b)?(a):(b))
#define MAXN (2000+10)
#define root (1)
#define INF (9999999)
using namespace std;
struct Segment_Tree_2D{
struct treeY{
int max,min;
int L,R;
int mid;
};
typedef struct treeY treeY;
struct treeX{
treeY T[MAXN];
int L,R;
int mid;
};
typedef struct treeX treeX;
int x1,x2,y1,y2,val,max_ans,min_ans,x0,y0;
treeX t[MAXN];
void queryY(int f,int r,int fa,int p){//ok
if (r>t[fa].T[p].R) r=t[fa].T[p].R;
if (f<t[fa].T[p].L) f=t[fa].T[p].L;
if (f==t[fa].T[p].L&&r==t[fa].T[p].R){
max_ans=bigger(max_ans,t[fa].T[p].max);
min_ans=minner(min_ans,t[fa].T[p].min);
}else{
int mid=t[fa].T[p].mid;
if (r>mid) queryY(mid+,r,fa,RS(p));
if (f<=mid) queryY(f,mid,fa,LS(p));
}
}
void queryX(int f,int r,int p){//ok
if (r>t[p].R) r=t[p].R;
if (f<t[p].L) f=t[p].L;
if (f==t[p].L&&r==t[p].R){
queryY(y1,y2,p,root);
}else{
int mid=t[p].mid;
if (r>mid) queryX(mid+,r,RS(p));
if (f<=mid) queryX(f,mid,LS(p));
}
}
void modifyY(int fa,int p){//ok
if (y0==t[fa].T[p].L&&y0==t[fa].T[p].R){
t[fa].T[p].min=val;
t[fa].T[p].max=val;
}else{
int mid=t[fa].T[p].mid;
if (y0>mid) modifyY(fa,RS(p));
if (y0<=mid) modifyY(fa,LS(p));
}
t[fa].T[p].max=bigger(t[fa].T[LS(p)].max,t[fa].T[RS(p)].max);
t[fa].T[p].min=minner(t[fa].T[LS(p)].min,t[fa].T[RS(p)].min);
}
void maintainY(int fa,int p){//ok
if (y0==t[fa].T[p].L&&y0==t[fa].T[p].R){
t[fa].T[p].min=minner(t[LS(fa)].T[p].min,t[RS(fa)].T[p].min);
t[fa].T[p].max=bigger(t[LS(fa)].T[p].max,t[RS(fa)].T[p].max);
}else{
int mid=t[fa].T[p].mid;
if (y0>mid) maintainY(fa,RS(p));
if (y0<=mid) maintainY(fa,LS(p));
}
t[fa].T[p].max=bigger(t[fa].T[LS(p)].max,t[fa].T[RS(p)].max);
t[fa].T[p].min=minner(t[fa].T[LS(p)].min,t[fa].T[RS(p)].min);
}
void modifyX(int p){
if (x0==t[p].L&&x0==t[p].R){
modifyY(p,root);
}else{
int mid=t[p].mid;
if (x0>mid) modifyX(RS(p));
if (x0<=mid) modifyX(LS(p));
maintainY(p,root);
}
} void query(){max_ans=-INF;min_ans=INF;queryX(x1,x2,root);}
void modify(){modifyX(root);}
};
typedef struct Segment_Tree_2D ST2;
ST2 t;
int main()
{
int i,j,n,m,temp,q;
char cmd[];
scanf("%d%d",&n,&m);
for (i=;i<=n;i++){
for (j=;j<=m;j++){
scanf("%d",&temp);
t.x0=i; t.y0=j;
t.modify();
}
}
scanf("%d",&q);
for (i=;i<q;i++){
scanf("%s",cmd);
if (cmd[]=='q'){
scanf("%d%d%d%d",&t.x1,&t.y1,&t.x2,&t.y2);
t.query();
printf("%d %d\n",t.max_ans,t.min_ans);
}else{
scanf("%d%d",&t.x0,&t.y0);
t.modify();
}
}
return ;
}
数据结构习题 线段树&树状数组的更多相关文章
- CodeForces -163E :e-Government (AC自动机+DFS序+树状数组)
The best programmers of Embezzland compete to develop a part of the project called "e-Governmen ...
- NOIp 数据结构专题总结 (2):分块、树状数组、线段树
系列索引: NOIp 数据结构专题总结 (1) NOIp 数据结构专题总结 (2) 分块 阅:<「分块」数列分块入门 1-9 by hzwer> 树状数组 Binary Indexed T ...
- 数据结构--树状数组&&线段树--基本操作
随笔目的:方便以后对树状数组(BIT)以及基本线段树的回顾 例题链接:http://acm.hdu.edu.cn/showproblem.php?pid=1166 例题:hdu 1166 敌兵布阵 T ...
- HDU 1556 线段树或树状数组,插段求点
1.HDU 1556 Color the ball 区间更新,单点查询 2.题意:n个气球,每次给(a,b)区间的气球涂一次色,问最后每个气球各涂了几次. (1)树状数组 总结:树状数组是一个查 ...
- tyvj P1716 - 上帝造题的七分钟 二维树状数组区间查询及修改 二维线段树
P1716 - 上帝造题的七分钟 From Riatre Normal (OI)总时限:50s 内存限制:128MB 代码长度限制:64KB 背景 Background 裸体就意味着 ...
- hdu 4836 The Query on the Tree(线段树or树状数组)
The Query on the Tree Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Ot ...
- 【BZOJ3295】动态逆序对(线段树,树状数组)
[BZOJ3295]动态逆序对(线段树,树状数组) 题面 Description 对于序列A,它的逆序对数定义为满足iAj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的 ...
- ZJOI 2017 树状数组(线段树套线段树)
题意 http://uoj.ac/problem/291 思路 不难发现,九条カレン醬所写的树状数组,在查询区间 \([1,r]\) 的时候,其实在查询后缀 \([r,n]\) :在查询 \([l,r ...
- 「CodePlus 2017 11 月赛」Yazid 的新生舞会(树状数组/线段树)
学习了新姿势..(一直看不懂大爷的代码卡了好久T T 首先数字范围那么小可以考虑枚举众数来计算答案,设当前枚举到$x$,$s_i$为前$i$个数中$x$的出现次数,则满足$2*s_r-r > 2 ...
随机推荐
- Youth Is Not a Time of Life
Youth is not a time of life; it is a state of mind.青春不是年华,而是心境: It is not a matter of rosy cheeks, r ...
- Java+selenium自动化测试基础
Java+selenium maven配置 maven的配置,但还需要建立maven的本地库,修改apach-maven的setting.xml http://www.cnblogs.com/haoa ...
- 注册Asp4.0到iis
- influxDB选择类函数
1)TOP()函数 作用:返回一个字段中最大的N个值,字段类型必须是长整型或float64类型. 语法: SELECT TOP(<field_key>[,<tag_keys>] ...
- Qt里的原子操作QAtomicInteger
所谓原子操作,即一系列复杂的操作能一气呵成,中间不被其他的操作打断.这在多线程程序中尤其常见,但要实现这种功能,既要考虑程序的良好设计,又要关心特定平台的体系结构和相关编译器对原子特性的支持程度.所以 ...
- yum命令的实例
1) 自定义yum仓库:createrepo 2) 自定义repo文件 3) 使用yum命令安装httpd软件包(在这里需要强调一点,本身执行yum.repos.d时,文件里面是有自带的yum源的,需 ...
- 转:使用awk命令获取文本的某一行,某一列
1.打印文件的第一列(域) : awk '{print $1}' filename2.打印文件的前两列(域) : awk '{print ...
- vue 基础核心学习
<html> <body> <div id="app"> {{ message }} </div> <div id=" ...
- MapReduceTopK TreeMap
版权声明: https://blog.csdn.net/zhangxiango/article/details/33319281 MapReduce TopK统计加排序中介绍的TopK在mapredu ...
- 笔画宽度变化(C++和matlab算法)
最近一直在看工作方面的书籍,把论文的事情搁置了,之前承诺的贴代码的事一直拖.现在把代码整理发上来,只有核心部分的,都不是我写的,我是网上整理下载的,matlab代码的效果比较差. 全部文件网盘下载地址 ...