zoukankan      html  css  js  c++  java
  • HDU 6621 K-th Closest Distance(主席树+二分)

    题意

    对于大小为n的正整数数组a,有q次询问,每次询问给出整数数p、k,要求输出离p第k近的数字,强制在线。

    思路

    主席树维护区间 ([l, r]) 数字出现次数,二分答案。

    Code

    	#include <bits/stdc++.h>
    
    using namespace std;
    typedef long long ll;
    const int maxn = 1e5+10;
    
    struct node {
        int l, r, val;
    }tr[maxn*21];
    int root[maxn], tot, a[maxn], b[maxn];
    
    void build(int l, int r, int &x) {
        x = ++tot;
        if(l==r) return;
        int mid = l+r>>1;
        build(l, mid, tr[x].l); build(mid+1, r, tr[x].r);
    }
    void update(int l, int r, int x, int &y, int p, int val=1) {
        y = ++tot;
        tr[y] = tr[x];
        tr[y].val += val;
    //    cout << l << " " << r << " " <<  tr[y].val << endl;
        if(l==r) return;
        int mid = l+r>>1;
        if(p<=mid) update(l, mid, tr[x].l, tr[y].l, p, val);
        else update(mid+1, r, tr[x].r, tr[y].r, p, val);
    }
    int query(int l, int r, int x, int y, int L, int R) {
        if(L<=l && r<=R) return tr[y].val-tr[x].val;
        int mid=l+r>>1, ans=0;
        if(L<=mid) ans=query(l, mid, tr[x].l, tr[y].l, L, R);
        if(R>mid) ans+=query(mid+1, r, tr[x].r, tr[y].r, L, R);
        return ans;
    }
    int T, n, m;
    void read(int &x) {
        x=0; char ch, c=getchar();
        while(c<'0' || c>'9') ch=c, c=getchar();
        while(c>='0' && c<='9') x=x*10+c-'0', c=getchar();
        if(ch=='-') x=-x;
    }
    
    int main() {
    //    freopen("in.txt", "r", stdin);
        scanf("%d", &T);
        while(T--) {
            scanf("%d%d", &n, &m);
            tot=0;
            build(1, maxn, root[0]);
            for (int i=1; i<=n; ++i) {
                scanf("%d", a + i);
                b[i] = a[i];
            }
            sort(b+1, b+1+n);
            int cnt = unique(b+1, b+1+n)-b-1;
            for (int i=1; i<=n; ++i) {
                a[i] = lower_bound(b+1, b+cnt+1, a[i])-b+1;
                update(1, maxn, root[i-1], root[i], a[i]);
            }
            int ans=0;
            for (int L, R, P, K, i=1; i<=m; ++i) {
                scanf("%d%d%d%d", &L, &R, &P, &K);
                L^=ans, R^=ans, P^=ans, K^=ans;
                if(L>R) swap(L, R);
                int l=0, r=1e6+10;
                while(l<r) {
                    int mid=l+r>>1;
                    int ll = lower_bound(b+1, b+1+cnt, P-mid)-b+1;
                    int rr = upper_bound(b+1, b+1+cnt, P+mid)-b;
                    int sum = query(1, maxn, root[L-1], root[R], ll, rr);
                    if(sum >= K) r=mid;
                    else l=mid+1;
                }
                printf("%d
    ", l);
                ans = l;
            }
        }
        return 0;
    }
    
  • 相关阅读:
    转:js中javascript:void(0) 真正含义
    Chrome Capabilities & ChromeOptions
    scrapy
    远离DoS攻击 Windows Server 2016发布DNS政策
    windows server 2012 AD 活动目录部署系列(五)备份和还原域控制器
    windows server 2012 AD 活动目录部署系列(七)Active Directory 的授权还原
    对AD域进行定期自动备份设置图解
    教程:使用Diskpart创建、扩展或删除磁盘分区
    虚拟化天花板将近,后虚拟化时代如何应对?
    图样图森破 设置虚拟机优先级真的很容易?
  • 原文地址:https://www.cnblogs.com/acerkoo/p/11305368.html
Copyright © 2011-2022 走看看