赞
踩
本质上仍然是一棵线段树,但它和每个节点用来表示一个区间内元素出现的次数,可以理解为维护区间的值域。
现有序列:4 2 1 3
线段树初始状态各个节点权值是
0
{0}
0
依次插入元素
4
{4}
4:
依次插入元素
2
{2}
2:
依次插入元素 1 {1} 1:
依次插入元素
3
{3}
3:
权值线段树维护的是桶,按值域开空间,维护的是个数。
简单线段树维护的是信息,按个数可开空间,维护的是特定信息。
权值线段树可以解决 数列第 k 大/小的问题
注意:我们只能对给定数列解决整个数列的第k大/小,并不能解决数列的子区间的第k大/小。
建树
void build(int u, int l, int r) {
tr[u] = {l, r};
if(l == r) { tr[u].sum = a[l]; return ; }
int mid = l + r >> 1;
build(ls, l, mid), build(rs, mid + 1, r);
pushup(u);
}
修改
void modify(int u, int x) {
tr[u].sum ++;
if(tr[u].l == tr[u].r) return ;
int mid = tr[u].l + tr[u].r >> 1;
if(x <= mid) modify(ls, x);
else modify(rs, x);
pushup(u);
}
询问
int query(int u, int k) {
if(tr[u].l == tr[u].r) return tr[u].l;
if(tr[ls].sum >= k) return query(ls, k);
return query(rs, k - tr[ls].sum);
}
现有 n {n} n 个正整数,要求出这 n {n} n 个正整数中的第 k {k} k 个最小整数 (相同大小的整数只计算一次)。
第一行为 n {n} n 和 k {k} k ; 第二行开始为 n {n} n 个正整数的值,整数间用空格隔开。
第
k
{k}
k个最小整数的值;若无解,则输出 NO RESULT
。
权值线段树,维护桶 a {a} a。
建空树,加修改
#include <bits/stdc++.h> using namespace std; const int N = 100010; #define ls u<<1 #define rs u<<1|1 struct node { int l, r, sum; } tr[N << 2]; int n, k; int a[N]; void pushup(int u) { tr[u].sum = tr[ls].sum + tr[rs].sum; } void build(int u, int l, int r) { tr[u] = {l, r}; if(l == r) return ; int mid = l + r >> 1; build(ls, l, mid), build(rs, mid + 1, r); pushup(u); } void modify(int u, int x) { tr[u].sum ++; if(tr[u].l == tr[u].r) return ; int mid = tr[u].l + tr[u].r >> 1; if(x <= mid) modify(ls, x); else modify(rs, x); pushup(u); } int query(int u, int k) { if(tr[u].l == tr[u].r) return tr[u].l; if(tr[ls].sum >= k) return query(ls, k); return query(rs, k - tr[ls].sum); } int main() { cin >> n >> k; int mx = 0; for (int i = 0; i < n; i++) { int x; cin >> x; a[x] = 1; mx = max(mx, x); } build(1, 1, mx); for (int i = 0; i <= mx; i++) if(a[i]) modify(1, i); if(k > tr[1].sum) cout << "NO RESULT" << endl; else cout << query(1, k) << endl; return 0; }
#include <bits/stdc++.h> using namespace std; const int N = 100010; #define ls u<<1 #define rs u<<1|1 struct node { int l, r, sum; } tr[N << 2]; int n, k; int a[N]; void pushup(int u) { tr[u].sum = tr[ls].sum + tr[rs].sum; } void build(int u, int l, int r) { tr[u] = {l, r}; if(l == r) { tr[u].sum = a[l]; return ; } int mid = l + r >> 1; build(ls, l, mid), build(rs, mid + 1, r); pushup(u); } int query(int u, int k) { if(tr[u].l == tr[u].r) return tr[u].l; if(tr[ls].sum >= k) return query(ls, k); return query(rs, k - tr[ls].sum); } int main() { cin >> n >> k; int mx = 0; for (int i = 0; i < n; i++) { int x; cin >> x; a[x] = 1; mx = max(mx, x); } build(1, 1, mx); if(k > tr[1].sum) cout << "NO RESULT" << endl; else cout << query(1, k) << endl; return 0; }
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。