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 ...
随机推荐
- Conetos 下安装docker 和镜像加速
首先升级yum yum update 安装yum-utils,它提供yum-config-manager可以用来配置repo yum install -y yum-utils 使用以下命令设置稳定版 ...
- Linux文件目录的权限
权限对文件的重要性:(主要是针对文件的内容而言,与文件名没有关系) r: 可读取此文件的实际内容. w: 可以编辑.新增或者修改该文件的内容(但不能删除该文件) x: 该文件具有可以被系统执行的权限. ...
- 未能加载文件或程序集“microsoft.Build.Engine, Version=3.5.0.0,...”或它的摸一个依赖项。
今天想打开IIS服务,然后点错了,不小心关掉了.net组件,结果vs就一直打不开项目,最后在网上查到了原因,打开 控制面板->程序和功能->打开或关闭功能 在里面勾选Microsoft . ...
- Statistics项目学习笔记
1. http://218.244.157.0:55443/index.html 初始访问时,弹出的窗口为index.html文件,文件有html命令组成.html展现的UI界面用的是WIN10-UI ...
- web框架 Spring+SpringMvc+Jpa 纯注解搭建
0.jar包依赖 maven pom.xml <properties> <spring_version>4.3.25.RELEASE</spring_version&g ...
- noip2013day1-货车运输
题目描述 \(A\)国有\(n\)座城市,编号从 \(1\)到\(n\),城市之间有 \(m\) 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 \(q\) 辆货车在运输货物, 司机们想知 ...
- Ubuntu下火狐浏览器播放视频出现解码问题
问题描述 点击视频播放按钮,视频不会出现缓冲条,也没任何提示,视频界面就一直是黑屏的状态.右键该视频界面,选择检查元素,点击控制台,发现如下问题: The video on this page can ...
- a页面通过url传值,b页面如何接收(jquery.params.js实现)
用于两个html页面之间的传值 我的应用场景是:用echarts在a页面做完中国地图后,点击某个省份在b页面显示某个省份的地图.(在b页面显示点击了的那个省份的地图,等于说b页面是个“容器”页) 假设 ...
- Vue 基础语法入门(转载)
使用vue.js原文介绍:Vue.js是一个构建数据驱动的web界面库.Vue.js的目标是通过尽可能简单的API实现响应式数据绑定和组合的视图组件.vue.js上手非常简单,先看看几个例子: 例一: ...
- 解决Linux下SSH超时自动断开
title: 解决Linux下SSH超时自动断开 comments: false date: 2019-08-19 19:22:55 description: Linux 下 SSH 超时自动断开?? ...