正题
题目链接:https://www.luogu.com.cn/problem/P7011
题目大意
给出nnn个点的一棵树,从一出发,要走到 ttt。初始时权值为000,每个节点有一个权值wiw_iwi,第一次走过这个节点时会让权值加上该节点的权值,要求全程权值不能为负数,求能否走到ttt。
1≤n≤2×1051\leq n\leq 2\times 10^51≤n≤2×105
解题思路
第一个比较麻烦的点是有一个终点的限制,我们走到终点之后就不需要考虑其他点了,不妨在终点后接一个权值为+∞+\infty+∞的节点,然后问题变为能否遍历全树。
接下来我们会发现有个很麻烦的点,因为对于一个子树我们可能进入多次,一个比较暴力的想法是我们可以设fx,jf_{x,j}fx,j表示我们在权值为jjj的时候进入xxx的子树再出来时能够变为的最大权值。
假设我们权值从jjj增加到了j+kj+kj+k,那么再进入能够获得的贡献就是fi,j+k−fi,jf_{i,j+k}-f_{i,j}fi,j+k−fi,j,不难发现对于一个fi,jf_{i,j}fi,j不同的权值个数只有最多子树大小个。考虑维护这些变换的位置,记为若干个二元组(x,y)(x,y)(x,y)表示权值为xxx时进入能够获得yyy的权值,显然的我们有这些区间[x,x+y][x,x+y][x,x+y]是不相交的(因为如果相交那么可以一次获得更多权值)。
而合并的时候我们直接暴力把这些区间合并(因为即使表现上相交了,我们可以后续考虑节点权值的时候再合并这些区间),然后根据节点xxx的权值暴力合并前面的区间。
至于两个堆的合并自然可以用可并堆但是不如启发式合并好写。
时间复杂度:O(nlog2n)O(n\log^2 n)O(nlog2n)
code
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
#define ll long long
#define mp(x,y) make_pair(x,y)
using namespace std;
const ll N=2e5+10;
struct node{ll to,next;
}a[N<<1];
ll T,n,t,tot,ls[N],p[N],w[N];
priority_queue<pair<ll,ll> >q[N];
void addl(ll x,ll y){a[++tot].to=y;a[tot].next=ls[x];ls[x]=tot;return;
}
void dfs(ll x,ll fa){p[x]=x;for(ll i=ls[x];i;i=a[i].next){ll y=a[i].to;if(y==fa)continue;dfs(y,x);if(q[p[y]].size()>q[p[x]].size())swap(p[x],p[y]);while(!q[p[y]].empty()){q[p[x]].push(q[p[y]].top());q[p[y]].pop();}}pair<ll,ll> k=mp(0,w[x]);while(!q[p[x]].empty()){pair<ll,ll> z=q[p[x]].top();z.first=-z.first;if(k.second>=0&&z.first>k.first+k.second)break;k=mp(max(k.first,z.first-k.second),k.second+z.second);q[p[x]].pop();}k.first=-k.first;if(k.second>0)q[p[x]].push(k);return;
}
signed main()
{scanf("%lld",&T);while(T--){scanf("%lld%lld",&n,&t);tot=0;for(ll i=1;i<=n+1;i++){ls[i]=0;while(!q[i].empty())q[i].pop();}for(ll i=1;i<=n;i++)scanf("%lld",&w[i]);for(ll i=1,x,y;i<n;i++){scanf("%lld%lld",&x,&y);addl(x,y);addl(y,x);}++n;w[n]=1e18;addl(t,n);addl(n,t);dfs(1,0);if(q[p[1]].empty())puts("trapped");else{pair<ll,ll> w=q[p[1]].top();if(w.first>=0&&w.second>1e17)puts("escaped");else puts("trapped");}}return 0;
}