主席树-----动态开点,不hash
第k大
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath> #include <vector>
#include <iostream>
#include <algorithm>
using namespace std; const int maxn = + ; struct Node {
int L, R;
int sum;
} seg[maxn << ]; int T[maxn]; int tot; int build(int L, int R) {
int root = ++tot;
seg[root].sum = ;
if (L < R) {
int mid = (L + R) >> ;
seg[root].L = build(L, mid);
seg[root].R = build(mid + , R);
}
return root;
} int upDate(int version, int L, int R, int val) {
int root = ++tot;
seg[root] = seg[version];
seg[root].sum = seg[version].sum + ;
if (L < R) {
int mid = (L + R) >> ;
if (val <= mid) seg[root].L = upDate(seg[version].L, L, mid, val);
else seg[root].R = upDate(seg[version].R, mid + , R, val);
}
return root;
} int query(int u, int v, int L, int R, int k) {
if (L >= R) return L;
int num = seg[seg[v].L].sum - seg[seg[u].L].sum;
int mid = (L + R) >> ;
if (num >= k) {
return query(seg[u].L, seg[v].L, L, mid, k);
} else return query(seg[u].R, seg[v].R, mid + , R, k - num);
} int a[maxn];
vector<int> vc; int getID(int val) {
return lower_bound(vc.begin(), vc.end(), val) - vc.begin() + ;
} const int down = -1e9, up = 1e9; void work() {
tot = ;
vc.clear();
int n, m;
cin >> n >> m;
for (int i = ; i <= n; ++i) {
scanf("%d", a + i);
vc.push_back(a[i]);
}
sort(vc.begin(), vc.end());
vc.erase(unique(vc.begin(), vc.end()), vc.end());
// for (int i = 0; i < vc.size(); ++i) {
// printf("%d ", vc[i]);
// }
T[] = build(up, down);
for (int i = ; i <= n; ++i) {
// int id = getID(a[i]);
T[i] = upDate(T[i - ], down, up, a[i]);
}
for (int i = ; i <= m; ++i) {
int L, R, val;
scanf("%d%d%d", &L, &R, &val);
// val = (R - L + 1) - val + 1;
int res = query(T[L - ], T[R], down, up, val);
printf("%d\n", res);
} } int main() {
#ifdef local
freopen("data.txt", "r", stdin);
#endif // local
// int t;
// scanf("%d", &t);
// while (t--) work();
work();
return ;
}
区间不同元素个数
https://vjudge.net/problem/SPOJ-DQUERY
#include <bits/stdc++.h>
using namespace std;
const int inf = 0x3f3f3f3f; const int maxn = + ; struct Node {
int L, R;
int sum;
} seg[maxn << ];
int T[maxn], now; int build(int L, int R) {
++now;
seg[now].sum = ;
if (L < R) {
int mid = (L + R) >> ;
seg[now].L = build(L, mid);
seg[now].R = build(mid + , R);
}
return now;
} int upDate(int version, int L, int R, int val, int xo) {
int root = ++now;
seg[root] = seg[version];
seg[root].sum = seg[version].sum + xo;
if (L < R) {
int mid = (L + R) >> ;
if (val <= mid) {
seg[root].L = upDate(seg[version].L, L, mid, val, xo);
} else {
seg[root].R = upDate(seg[version].R, mid + , R, val, xo);
}
}
return root;
} int query(int version1, int version2, int L, int R, int be, int en) {
if (L > R) return ;
if (L >= be && R <= en) {
return seg[version2].sum - seg[version1].sum;
}
int ans = ;
int mid = (L + R) >> ;
if (mid >= be) ans += query(seg[version1].L, seg[version2].L, L, mid, be, en);
if (mid + <= en) ans += query(seg[version1].R, seg[version2].R, mid + , R, be, en);
return ans;
} int pre[ + ]; int up = -1e9;
int en = 1e9; void work() {
int n;
scanf("%d", &n);
for (int i = ; i <= n; ++i) {
int x;
scanf("%d", &x);
if (pre[x]) {
int res = upDate(T[i - ], up, en, pre[x], -);
T[i] = upDate(res, up, en, i, );
} else T[i] = upDate(T[i - ], up, en, i, );
pre[x] = i;
}
int q;
cin >> q;
while (q--) {
int l, r;
scanf("%d%d", &l, &r);
printf("%d\n", query(T[l - ], T[r], up, en, l, r));
}
} int main() {
#ifdef local
freopen("data.txt", "r", stdin);
#endif // local
work();
return ;
}
主席树-----动态开点,不hash的更多相关文章
- 主席树--动态区间第k小
主席树--动态区间第\(k\)小 模板题在这里洛谷2617. 先对几个问题做一个总结: 阅读本文需要有主席树的基础,也就是通过区间kth的模板题. 静态整体kth: sort一下找第k小,时间复杂度\ ...
- BZOJ_4636_蒟蒻的数列_线段树+动态开点
BZOJ_4636_蒟蒻的数列_线段树+动态开点 Description 蒟蒻DCrusher不仅喜欢玩扑克,还喜欢研究数列 题目描述 DCrusher有一个数列,初始值均为0,他进行N次操作,每次将 ...
- hdu6183 Color it 线段树动态开点+查询减枝
题目传送门 题目大意: 有多次操作.操作0是清空二维平面的点,操作1是往二维平面(x,y)上放一个颜色为c的点,操作2是查询一个贴着y轴的矩形内有几种颜色的点,操作3退出程序. 思路: 由于查询的矩形 ...
- P3939 数颜色 线段树动态开点
P3939 数颜色 线段树动态开点 luogu P3939 水.直接对每种颜色开个权值线段树即可,注意动态开点. #include <cstdio> #include <algori ...
- HDU - 6183 暴力,线段树动态开点,cdq分治
B - Color itHDU - 6183 题目大意:有三种操作,0是清空所有点,1是给点(x,y)涂上颜色c,2是查询满足1<=a<=x,y1<=b<=y2的(a,b)点一 ...
- 洛谷P3313 [SDOI2014]旅行 题解 树链剖分+线段树动态开点
题目链接:https://www.luogu.org/problem/P3313 这道题目就是树链剖分+线段树动态开点. 然后做这道题目之前我们先来看一道不考虑树链剖分之后完全相同的线段树动态开点的题 ...
- codedecision P1113 同颜色询问 题解 线段树动态开点
题目描述:https://www.cnblogs.com/problems/p/11789930.html 题目链接:http://codedecision.com/problem/1113 这道题目 ...
- 学习笔记--函数式线段树(主席树)(动态维护第K极值(树状数组套主席树))
函数式线段树..资瓷 区间第K极值查询 似乎不过似乎划分树的效率更优于它,但是如果主席树套树状数组后,可以处理动态的第K极值.即资瓷插入删除,划分树则不同- 那么原理也比较易懂: 建造一棵线段树(权值 ...
- ZOJ 2112 Dynamic Rankings(主席树の动态kth)
题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=2112 The Company Dynamic Rankings ...
随机推荐
- 数据结构 merge_link合并链表
问题描述 本题任务是维护一条非递减的链表,初始长度为 0,记这条链表为主链表.对主链表做 N 次操作,操作分两种:1 k a1 a2 … ak,表示一条长度为 k 且非递减的链表,需要将这条链表合并到 ...
- [学习笔记]_exit和exit深入理解
#include<stdio.h> #include<stdlib.h> #include<string.h> #include <unistd.h> ...
- Timer的schedule和scheduleAtFixedRate方法的区别解析(转)
在java中,Timer类主要用于定时性.周期性任务 的触发,这个类中有两个方法比较难理解,那就是schedule和scheduleAtFixedRate方法,在这里就用实例分析一下 (1)sched ...
- C++新标准:列表初始化
一.列表初始化意义 C++新标准为vector提供了一种新的初始化方式:列表初始化.适用于知道多个成员具体值的情况. 二.列表初始化用法 /*1.空vector<int>*/ vector ...
- vue2.x学习笔记
1.使用模板template的时候必须要有跟节点,可以支持表达式,但不支持正则,想使用正则就用过滤器. 2.数据在显示的时候所带的HTML DOM直接显示,不会渲染,要渲染DOM,得用v-html. ...
- Iterator 遍历器
1.遍历器(Iterator)是一种接口,为各种不同的数据结构提供统一的访问机制.任何数据结构只要部署Iterator接口,就可以完成遍历操作(即依次处理该数据结构的所有成员). 2.Iterator ...
- 转载文章 MySQL与Oracle的区别
MySQL与Oracle的区别 1. Oracle是大型数据库而Mysql是中小型数据库,Oracle市场占有率达40%,Mysql只有20%左右,同时Mysql是开源的而Oracle价格非常高 ...
- 数据结构4:顺序表(线性表的顺序存储结构)及C语言实现
逻辑结构上呈线性分布的数据元素在实际的物理存储结构中也同样相互之间紧挨着,这种存储结构称为线性表的顺序存储结构. 也就是说,逻辑上具有线性关系的数据按照前后的次序全部存储在一整块连续的内存空间中,之间 ...
- bootstrap-table 选择行,并且获得选中行的所有数据内容
html代码如下: <table id="table" data-toggle="table" th:attr="data-url=@{/vie ...
- C++_静态类成员
在C++中,静态成员是属于整个类的而不是某个对象. 静态成员变量只存储一份供所有对象共用,所以在所有对象中都可以共享它. 使用静态成员变量实现多个对象之间的数据共享不会破坏隐藏的原则,保证了安全性还可 ...