题解 P4197 【Peaks】

首先用\(Kruskal\)重构树,这样我们就能很方便的处理比\(x\)小的限制条件了

然后它询问的是高度中的第\(k\)大,会自然地想到要用主席树,只是说,要注意一个细节。

不难发现,重构树的某一个非叶节点(边)会管辖某些叶子节点(点),有趣的是,其管辖的叶子节点构成一个区间,换而言之是连续的(想不出来的可以画一张图\(QAQ\)

所以我可以先\(Kruskal\)重构树,然后做一遍\(dfs\),这遍\(dfs\)的过程中,我们把所有的叶子节点 "摘" 下来,形成一个序列,并对每个节点维护其管辖的区间。

然后每次询问就变成了:先在重构树上跳,跳到某个节点\(x\)后询问\(L[x]\)\(R[x]\)这一段区间内第\(k\)大的数是谁?—主席树板子

然后注意离散化,以及特判无解的情况。下面是丑陋的代码。


#include<bits/stdc++.h>
using namespace std;
int read(){
    char cc = getchar(); int cn = 0, flus = 1;
    while(cc < '0' || cc > '9'){
        if(cc == '-') flus = -flus;
        cc = getchar();
    }
    while(cc >= '0' && cc <= '9')
        cn = cn * 10 + cc - '0', cc = getchar();
    return cn * flus;
}
const int N = 2e5 + 5;
const int M = 5e5 + 5;
const int inf = 999999999;
struct E{
    int to, next;
}e[N * 2];
struct Node{
    int x, y, z;
}q[M], hh[N];
struct Tree{
    int l, r, val;
}t[N * 27];
int h[N], head[N], fa[N], val[N], fath[N][24], L[N], R[N], b[N];
int n, m, Q, cnt, tot, pop, rot[N], ctt;
bool cmp(Node x, Node y){
    return x.z < y.z;
}
void add(int x, int y){
    e[++cnt] = (E){y, head[x]}; head[x] = cnt;
}
void input(){
    n = read(), m = read(), Q = read();
    val[0] = inf;
    for(int i = 1; i <= n; i++) h[i] = read(), hh[i].x = i, hh[i].z = h[i];
    sort(hh + 1, hh + n + 1, cmp);
    for(int i = 1; i <= n; i++) h[hh[i].x] = i;
    for(int i = 1; i <= m; i++)
        q[i].x = read(), q[i].y = read(), q[i].z = read();
    return ;
}
int find(int x){
    if(fa[x] == x) return x;
    return fa[x] = find(fa[x]);
}
void Kruskal()
{
    sort(q + 1, q + m + 1, cmp);
    for(int i = 1; i <= n; i++) fa[i] = i;
    tot = n;
    for(int i = 1; i <= m; i++)
    {
        int u = find(q[i].x), v = find(q[i].y);
        if(u == v) continue;
        val[++tot] = q[i].z, fa[tot] = fa[u] = fa[v] = tot;
        add(tot, u), add(tot, v);
    }
}
void dfs(int x, int fa)
{
    fath[x][0] = fa;
    for(int i = 1; i <= 19; i ++)  
        fath[x][i] = fath[fath[x][i - 1]][i - 1];
    L[x] = pop;
    if(head[x] == 0) { b[++pop] = x; R[x] = pop; return ; }
    for(int i = head[x]; i; i = e[i].next)
        dfs(e[i].to, x);
    R[x] = pop;
}
//主席树 
void build(int &root, int ll, int rr)
{
    root = ++ctt;
    if(ll == rr)  return ;
    int mid = (ll + rr) >> 1;
    build(t[root].l, ll, mid); build(t[root].r, mid + 1, rr);
}
void change(int &root, int node, int ll, int rr, int k)
{
    root = ++ctt; t[root] = t[node];
    if(ll == rr) {  t[root].val ++;  return ; }
    int mid = (ll + rr) >> 1;
    if(mid >= k) change(t[root].l, t[node].l, ll, mid, k);
    else change(t[root].r, t[node].r, mid + 1, rr, k);
    t[root].val = t[t[root].l].val + t[t[root].r].val;
}
int find_tr(int x, int k)
{
    int now = x;
    for(int i = 19; i >= 0; i--)
        if(val[fath[now][i]] <= k) now = fath[now][i];
    return now;
}
int query(int root1, int root2, int k, int l, int r)
{
    int rkid = t[t[root1].r].val - t[t[root2].r].val, mid = (l + r) >> 1;
    if(l == r){
        if(k - (t[root1].val - t[root2].val) == 0) //这里注意特判
            return l;
        return 0;
    }
    if(rkid >= k) return query(t[root1].r, t[root2].r, k, mid + 1, r);
    else return query(t[root1].l, t[root2].l, k - rkid, l, mid);
}
void solve(){
    Kruskal(), dfs(tot, tot);
    build(rot[0], 1, n);
    for(int i = 1; i <= pop; i++)
        change(rot[i], rot[i - 1], 1, n, h[b[i]]);
    int v, x, k;
    hh[0].z = -1;
    for(int i = 1; i <= Q; i++)
    {
        v = read(), x = read(), k = read();
        int vip = find_tr(v, x);
        int ans = query(rot[R[vip]], rot[L[vip]], k, 1, n);
        printf("%d\n", hh[ans].z);
    }
}
signed main()
{
    input();
    solve();
    return 0;
}

转载于:https://www.cnblogs.com/Soulist/p/10587952.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值