SPOJ D-query && HDU 3333 Turing Tree (线段树 && 区间不相同数个数or和 && 离线处理)
题意 : 给出一段n个数的序列,接下来给出m个询问,询问的内容SPOJ是(L, R)这个区间内不同的数的个数,HDU是不同数的和
分析 :
一个经典的问题,思路是将所有问询区间存起来,然后按右端点排序
最后从左到右将原区间扫一遍,扫的过程当中不断将重复出现的数字右移
也就是如果一个数字重复出现,那么我们记录最右边的那个为有效的,其他都视为不存在
这样一旦遇到一个问询区间的右端点就线段树查询即可。
SPOJ:
#include<bits/stdc++.h>
using namespace std;
#define lson l , m , rt << 1
#define rson m + 1 , r , rt << 1 | 1
;
];
void PushUP(int rt) {
sumv[rt] = sumv[rt<<] + sumv[rt<<|];
}
void update(int p,int sc,int l,int r,int rt) {
if (l == r) {
sumv[rt] += sc;
return ;
}
);
if (p <= m) update(p , sc , lson);
else update(p , sc , rson);
PushUP(rt);
}
long long query(int L,int R,int l,int r,int rt) {
if (L <= l && r <= R) {
return sumv[rt];
}
);
;
if (L <= m) ret += query(L , R , lson);
if (R > m) ret += query(L , R , rson);
return ret;
}
struct Interval{
int l, r, id;
bool operator < (const Interval& other) const{
return this->r < other.r;
}
};
Interval sec[];
map<int, int> mp;
int arr[maxn];
int pre[maxn];
long long ans[maxn];
int main(void)
{
, m;
scanf("%d", &n);
; i<=n; i++){
scanf("%d", &arr[i]);
if(!mp.count(arr[i]))
mp[arr[i]] = cnt++;//区间上的每一个数都有一个独特的编号,使用map来进行映射,目的是方便判断此数是否已经重复出现过
}
scanf("%d", &m);
; i<=m; i++){
scanf("%d %d", &sec[i].l, &sec[i].r);
sec[i].id = i;//因为后续有排序操作,而我们又要对问询答案顺序给出,所以记录给出的顺序,方便输出答案
}
sort(sec+, sec++m);//按照右端点排序
bool vis[maxn];
memset(vis, false, sizeof(vis));
, j=; j<=m && i<=n; i++){
int tmp = mp[arr[i]];//拿出这个数的编号
if(vis[tmp]){//如果这个数在前面已经访问过
update(pre[tmp], -, , n, );//将之前的位置信息抹去
update(i, , , n, );//当前的位置才是有效的,所以给当前位置+1
pre[tmp] = i;//更新pre数组方便下次操作
}else{
vis[tmp] = true;
update(i, , , n, );
pre[tmp] = i;
}
while(sec[j].r == i){
ans[sec[j].id] = query(sec[j].l, sec[j].r, , n, );
j++;
}
}
; i<=m; i++){
printf("%I64d\n", ans[i]);
}
;
}
HDU:
#include<bits/stdc++.h>
using namespace std;
#define lson l , m , rt << 1
#define rson m + 1 , r , rt << 1 | 1
;
];
void PushUP(int rt) {
sumv[rt] = sumv[rt<<] + sumv[rt<<|];
}
void update(int p,long long sc,int l,int r,int rt) {
if (l == r) {
sumv[rt] += sc;
return ;
}
);
if (p <= m) update(p , sc , lson);
else update(p , sc , rson);
PushUP(rt);
}
long long query(int L,int R,int l,int r,int rt) {
if (L <= l && r <= R) {
return sumv[rt];
}
);
;
if (L <= m) ret += query(L , R , lson);
if (R > m) ret += query(L , R , rson);
return ret;
}
struct interval{
int L, R, id;
bool operator < (const interval & other) const{
return this->R < other.R;
}
};
interval sec[maxn<<];
bool vis[maxn];
];
int per[maxn];
long long arr[maxn];
map<long long, int> mp;
int main(void)
{
int nCase;
scanf("%d", &nCase);
while(nCase--){
int n;
mp.clear(); ;
memset(sumv, , sizeof(sumv));
scanf("%d", &n);
; i<=n; i++){
scanf("%I64d", &arr[i]);
if(!mp.count(arr[i]))
mp[arr[i]] = cnt++;
}
int m;
scanf("%d", &m);
; i<=m; i++){
scanf("%d %d", &sec[i].L, &sec[i].R);
sec[i].id = i;
}
sort(sec+, sec++m);
memset(vis, false, sizeof(vis));
, j=; j<=m && i<=n; i++){
int num = mp[arr[i]];
if(vis[num]){
update(per[num],-arr[i],,n,);
update(i, arr[i], , n, );
per[num] = i;
}else{
vis[num] = true;
update(i, arr[i], , n, );
per[num] = i;
}
while(i==sec[j].R){
ans[sec[j].id] = query(sec[j].L, sec[j].R, , n, );
j++;
}
}
; i<=m; i++){
printf("%I64d\n", ans[i]);
}
}
}
update in 2018-06-08
学了主席树、以上的离线算法都可以变成在线算法了、戳 主席树
SPOJ D-query && HDU 3333 Turing Tree (线段树 && 区间不相同数个数or和 && 离线处理)的更多相关文章
- HDU 3333 Turing Tree 线段树+离线处理
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3333 Turing Tree Time Limit: 6000/3000 MS (Java/Othe ...
- HDU 3333 Turing Tree (线段树)
Turing Tree Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Tota ...
- HDU 3333 Turing Tree(树状数组/主席树)
题意 给定一个长度为 \(n\) 的序列,\(m\) 个查询,每次查询区间 \([L,R]\) 范围内不同元素的和. \(1\leq T \leq 10\) \(1 \leq n\leq 300 ...
- HDU 3333 Turing Tree (主席树)
题意:给定上一个序列,然后有一些询问,求区间 l - r 中有多少个不同的数的和. 析:和求区间不同数的方法是一样,只要用主席树维护就好. 代码如下: #pragma comment(linker, ...
- SPOJ GSS1_Can you answer these queries I(线段树区间合并)
SPOJ GSS1_Can you answer these queries I(线段树区间合并) 标签(空格分隔): 线段树区间合并 题目链接 GSS1 - Can you answer these ...
- W同学的新画板 QDUOJ 线段树 区间颜色段数
W同学的新画板 QDUOJ 线段树 区间颜色段数 原题链接 题意 W同学在每天的刻苦学习完成功课之余,都会去找一些有趣的事情来放松自己:恰巧今天他收到了朋友送给他的一套画板,于是他立刻拆开了包装,拿出 ...
- hdu 3333 Turing Tree 图灵树(线段树 + 二分离散)
http://acm.hdu.edu.cn/showproblem.php?pid=3333 Turing Tree Time Limit: 6000/3000 MS (Java/Others) ...
- hdu 3333 Turing Tree(线段树+离散化)
刚看到是3xian大牛的题就让我菊花一紧,觉着这题肯定各种高端大气上档次,结果果然没让我失望. 刚开始我以为是一个普通的线段树区间求和,然后啪啪啪代码敲完测试没通过,才注意到这个求和是要去掉相同的值的 ...
- HDU 6638 - Snowy Smile 线段树区间合并+暴力枚举
HDU 6638 - Snowy Smile 题意 给你\(n\)个点的坐标\((x,\ y)\)和对应的权值\(w\),让你找到一个矩形,使这个矩阵里面点的权值总和最大. 思路 先离散化纵坐标\(y ...
随机推荐
- 小记---------Elasticsear搭建
Elasticsear搭建 创建用户: useradd elasticsearch passwd elasticsearch 1.解压 tar -zxvf elasticsearch-5.5.2. ...
- CMDB 理论
TIL即IT基础架构库(Information Technology Infrastructure Library, ITIL,信息技术基础架构库)由英国政府部门CCTA(Central Comput ...
- 【五一qbxt】test1
(不知道为什么居然爆零了qwq) (全员爆零诶,最高分10分???还是rand出来的???) 我freopen写错了????自闭了 不行不行再写一遍freopen加深印象,不能再写错了 freopen ...
- pandas中数据结构-Series
pandas中数据结构-Series pandas简介 Pandas是一个开源的,BSD许可的Python库,为Python编程语言提供了高性能,易于使用的数据结构和数据分析工具.Python与Pan ...
- Luogu P2612 [ZJOI2012]波浪
题目 我们考虑从\(1\)到\(n\)把每个数放到序列里面去,以消掉绝对值. 在最后的序列中,如果\(i\)的某一边是序列的边界,那么\(i\)会产生\(0\)的贡献.如果\(i\)的某一边是一个比\ ...
- 声明一个LIst类型的数组
ArrayList[] graphArrayList = new ArrayList[4]; for(int i=0;i<graphArrayList.length;i++){ graphArr ...
- 使用ActiveMQ实现JMS消息通信服务
PTP(点对点的消息模型) 在点对点模型中,相当于两个人打电话,两个人独享一条通信线路.一方发送消息,一方接收消息. 在p2p的模型中,双方通过队列交流,一个队列只有一个生产者和一个消费者. 1.建立 ...
- 深入解析Vue.js中v-bind v-model的使用和区别
v-model 指令在表单控件元素上创建双向数据绑定,所谓双向绑定,指的就是我们在js中的vue实例中的data与其渲染的dom元素上的内容保持一致,两者无论谁被改变,另一方也会相应的更新为相同的数据 ...
- maven配置生成可执行的jar:maven-shade-plugin
默认打包生成的jar是不能直接运行的,因为带有main方法的信息不会添加到mainifest中,需要借助maven-shade-plugin <project> ... <build ...
- Mac中如何高亮显示当前git分支以及相关切换系统默认shell操作
前言 在开发中对于使用git的同学,肯定会需要高亮显示当前分支,我们今天就来说一种最简单的办法,先来看一下效果图 实现 1.安装item2 https://www.iterm2.com 2.安装oh- ...