#2284. 接水果(fruit)

题意
内存限制:512 MiB
时间限制:6000 ms

风见幽香非常喜欢玩一个叫做 osu! 的游戏,其中她最喜欢玩的模式就是接水果。由于她已经 DT FC 了 The big black,她觉得这个游戏太简单了,于是发明了一个更加难的版本。

首先有一个地图,是一棵由 $n$ 个顶点、$n-1$ 条边组成的树(例如图 $1$ 给出的树包含 $8$ 个顶点、$7$ 条边)。这颗树上有 P 个盘子,每个盘子实际上是一条路径(例如图 $1$ 中顶点 $6$ 到顶点 $8$ 的路径),并且每个盘子还有一个权值。第 $i$ 个盘子就是顶点 $a_i$ 到顶点 $b_i$ 的路径(由于是树,所以从 $a_i$ 到 $b_i$ 的路径是唯一的),权值为 $c_i$。接下来依次会有 $Q$ 个水果掉下来,每个水果本质上也是一条路径,第 $i$ 个水果是从顶点 $u_i$ 到顶点 $v_i$ 的路径。

幽香每次需要选择一个盘子去接当前的水果:一个盘子能接住一个水果,当且仅当盘子的路径是水果的路径的子路径(例如图 $1$ 中从 $3$ 到 $7$ 的路径是从 $1$ 到 $8$ 的路径的子路径)。这里规定:从 $a$ 到 $b$ 的路径与从 $b$ 到 $a$ 的路径是同一条路径。当然为了提高难度,对于第 $i$ 个水果,你需要选择能接住它的所有盘子中,权值第 $k_i$ 小的那个盘子,每个盘子可重复使用(没有使用次数的上限:一个盘子接完一个水果后,后面还可继续接其他水果,只要它是水果路径的子路径)。幽香认为这个游戏很难,你能轻松解决给她看吗?

对于所有数据,$N,P,Q leq 40000$。
题解
考虑整体二分
把二分到的值 $leq mid$ 的盘子加入到某数据结构,然后对于该二分区间内的询问查询其被水果覆盖的盘子个数和 $k$ 的关系,再分左右区间
考虑如何处理一条路径被另一条路径覆盖问题
先求出其 $dfs$ 序,设 $L_u=dfn_u$ ,$R_u=dfn_u+sz_u-1$
将水果看成二维平面上的点 $(L_u,L_v)$
若要覆盖路径 $(x,y)$ 则应该:

1. $Lca(x,y) != x$
也就是 $L_x leq L_u leq R_x$,$L_y leq L_v leq R_y$
所以可以把 $(x,y)$ 看成矩形 $(L_x,L_y),(R_x,R_y)$
2. $Lca(x,y)=x$
设 $z$ 是 $x→y$ 上的第一个点,
那也就是满足一点在 $y$ 子树内,一点在 $z$ 子树外
即 $(1,L_y),(L_z-1,R_y)$ 和 $(R_z+1,L_y),(n,R_y)$

这样就是二维数点问题啦,用 $bit$ 处理即可
(矩形记得横纵坐标相反也算

代码

#include <bits/stdc++.h>
#define I inline
using namespace std;
const int N=40005;
int m,n,P,Q,V[N*2],nx[N*2],hd[N],t,f[N][17],d[N],in[N],tt,b[N],B,ans[N],ot[N],s[N];
struct O1{int x,l,r,w,v;}p[N*8],pp[N*8];
struct O2{int x,y,k,id;}q[N],qq[N];
I void add(int u,int v){V[++t]=v;nx[t]=hd[u];hd[u]=t;}
I void dfs(int x,int fa){
    f[x][0]=fa;d[x]=d[fa]+1;in[x]=++tt;
    for (int i=1;f[f[x][i-1]][i-1];i++)
        f[x][i]=f[f[x][i-1]][i-1];
    for (int i=hd[x];i;i=nx[i])
        if (V[i]!=fa) dfs(V[i],x);ot[x]=tt;
}
I int get(int x,int y){
    int l=d[y]-d[x]-1;
    for (int i=16;~i;i--)
        if ((l>>i)&1) y=f[y][i];
    return y;
}
I void ins(int X1,int Y1,int X2,int Y2,int w){
    p[++m]=(O1){X1,X2,Y2,1,w},p[++m]=(O1){Y1+1,X2,Y2,-1,w},
    p[++m]=(O1){X2,X1,Y1,1,w},p[++m]=(O1){Y2+1,X1,Y1,-1,w};
}
I bool c1(O1 A,O1 B){return A.x<B.x;}
I bool c2(O2 A,O2 B){return A.x<B.x;}
I void update(int x,int v){for (;x<=n;x+=x&-x) s[x]+=v;}
I int query(int x){int A=0;for (;x;x-=x&-x) A+=s[x];return A;}
I void update(int l,int r,int v){update(l,v);update(r+1,-v);}
I void solve(int l,int r,int pl,int pr,int ql,int qr){
    if (l==r){
        for (int i=ql;i<=qr;i++)
            ans[q[i].id]=b[l];
        return;
    }
    int mid=(l+r)>>1,Pl=pl-1,Pr=pr+1,Ql=ql-1,Qr=qr+1,L;
    for (int k,j=pl,i=ql;i<=qr;i++){
        while(j<=pr && p[j].x<=q[i].x){
            if (p[j].v>b[mid]) pp[--Pr]=p[j];
            else
                update(p[j].l,p[j].r,p[j].w),
                pp[++Pl]=p[j];j++;
        }
        k=query(q[i].y);
        if (q[i].k>k) q[i].k-=k,qq[--Qr]=q[i];
        else qq[++Ql]=q[i];
        if (i==qr) while(j<=pr){
            if (p[j].v>b[mid]) pp[--Pr]=p[j];
            else update(p[j].l,p[j].r,p[j].w),pp[++Pl]=p[j];j++;
        }
    }
    for (int j=pl;j<=pr;j++)
        if (p[j].v<=b[mid]) update(p[j].l,p[j].r,-p[j].w);
    L=pl-1;for (int i=pl;i<=Pl;i++) p[++L]=pp[i];
    for (int i=pr;i>=Pr;i--) p[++L]=pp[i];
    L=ql-1;for (int i=ql;i<=Ql;i++) q[++L]=qq[i];
    for (int i=qr;i>=Qr;i--) q[++L]=qq[i];
    if (pl<=Pl && ql<=Ql) solve(l,mid,pl,Pl,ql,Ql);
    if (Pr<=pr && Qr<=qr) solve(mid+1,r,Pr,pr,Qr,qr);
}
int main(){
    scanf("%d%d%d",&n,&P,&Q);
    for (int x,y,i=1;i<n;i++)
        scanf("%d%d",&x,&y),
        add(x,y),add(y,x);dfs(1,0);
    for (int z,x,y,i=1;i<=P;i++){
        scanf("%d%d%d",&x,&y,&b[i]);
        if (d[x]>d[y]) swap(x,y);
        if (in[x]<=in[y] && ot[x]>=ot[y]){
            z=get(x,y);
            if (in[z]>1) ins(1,in[z]-1,in[y],ot[y],b[i]);
            if (ot[z]<n) ins(ot[z]+1,n,in[y],ot[y],b[i]);
        }
        else ins(in[x],ot[x],in[y],ot[y],b[i]);
    }
    sort(b+1,b+P+1);B=unique(b+1,b+P+1)-b-1;
    for (int x,y,i=1;i<=Q;i++)
        scanf("%d%d%d",&x,&y,&q[i].k),
        q[i].id=i,q[i].x=in[x],q[i].y=in[y];
    sort(p+1,p+m+1,c1);sort(q+1,q+Q+1,c2);
    solve(1,B,1,m,1,Q);
    for (int i=1;i<=Q;i++)
        printf("%d
",ans[i]);
    return 0;
}
原文地址:https://www.cnblogs.com/xjqxjq/p/10666872.html