考虑树是以1为中心的菊花图的情况,也即如何安排打怪兽的顺序

用二元组$(a,b)$来描述怪兽,则对于两个怪兽$(a_{1},b_{1})$和$(a_{2},b_{2})$,交换两者不会影响血量的变化量,而会改变初始血量的需求,具体即比较$\max(a_{1},a_{1}-b_{1}+a_{2})$和$\max(a_{2},a_{2}-b_{2}+a_{1})$

如果能证明传递性,那么以此进行排序即可

观察上述式子,可以得到以下信息:

1.$a\le b$的怪兽优于$a>b$的怪兽

2.对于$a\le b$的怪兽,$a$小的怪兽优于$a$大的怪兽

3.对于$a>b$的怪兽,$b$大的怪兽优于$b$小的怪兽

(代入均可证明)

进而上述信息足以比较,同时也具有传递性,即得证

回到原问题,有一个比较经典的套路:

考虑当前最优的位置,其父亲的怪兽被打死后一定会打其(注意怪兽只会被消除而不会增加),进而不妨将其与父亲合并(先打父亲再打其),合并方式参考之前

关于过程的维护,可以用一个并查集+set实现

时间复杂度为$o(n\log n)$,可以通过

 1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 100005
4 #define ll long long
5 struct Data{
6 ll a,b;
7 bool operator < (const Data &k)const{
8 if ((a<=b)!=(k.a<=k.b))return a<=b;
9 if (a<=b)return a<k.a;
10 return b>k.b;
11 }
12 }a[N];
13 vector<int>v[N];
14 set<pair<Data,int> >S;
15 int t,n,x,y,fa[N],f[N];
16 Data merge(Data x,Data y){
17 ll a=max(x.a,x.a-x.b+y.a);
18 return Data{a,x.b+y.b-x.a-y.a+a};
19 }
20 int find(int k){
21 if (k==f[k])return k;
22 return f[k]=find(f[k]);
23 }
24 void dfs(int k,int f){
25 fa[k]=f;
26 for(int i=0;i<v[k].size();i++)
27 if (v[k][i]!=f)dfs(v[k][i],k);
28 }
29 void merge(int x,int y){
30 x=find(x),y=find(y);
31 if (x!=1)S.erase(make_pair(a[x],x));
32 f[y]=x,a[x]=merge(a[x],a[y]);
33 if (x!=1)S.insert(make_pair(a[x],x));
34 }
35 int main(){
36 scanf("%d",&t);
37 while (t--){
38 scanf("%d",&n);
39 a[1]=Data{0,0},S.clear();
40 for(int i=1;i<=n;i++)v[i].clear();
41 for(int i=2;i<=n;i++){
42 scanf("%lld%lld",&a[i].a,&a[i].b);
43 S.insert(make_pair(a[i],i));
44 }
45 for(int i=1;i<n;i++){
46 scanf("%d%d",&x,&y);
47 v[x].push_back(y);
48 v[y].push_back(x);
49 }
50 dfs(1,0);
51 for(int i=1;i<=n;i++)f[i]=i;
52 for(int i=1;i<n;i++){
53 int x=(*S.begin()).second;
54 S.erase(S.begin());
55 merge(fa[x],x);
56 }
57 printf("%lld\n",a[1].a);
58 }
59 return 0;
60 }

最新文章

  1. java中文文档官方下载
  2. 深受C/C 程序员欢迎的11款IDE
  3. Linux(Ubuntu)之设定开机自启动
  4. PMP 第十一章 项目风险管理
  5. 43. Merge Sorted Array &amp;&amp; LRU Cache
  6. WCF使用泛型方法的问题
  7. Linux简介与厂商版本
  8. c#判断特殊字符?
  9. ajax 传参 乱码问题
  10. Ruby on Rails Tutorial 第四章 Rails背后的Ruby 之 字符串
  11. cmake 学习笔记(三)
  12. oracle行转列函数
  13. Spring BeanFactory源码学习
  14. 【JDK和Open JDK】平常使用的JDK和Open JDK有什么区别
  15. [福大软工] Z班——个人技术博客评分
  16. sqlserver触发器insert,delete,update
  17. DAY6 元组、字典与集合
  18. pika配置文件说明
  19. vue相关安装命令
  20. FastJson/spring boot: json输出

热门文章

  1. The Data Way Vol.3|做到最后只能删库跑路?DBA 能做的还有很多
  2. Python笔记_1语法总结
  3. Java(34)IO流之字符流
  4. Visual Studio Debug only user code with Just My Code
  5. 【实验向】问题:假设计算机A和计算机B通信,计算机A给计算机B发送一串16个字节的二进制字节串,以数组形式表示:
  6. 【数据结构与算法Python版学习笔记】树——利用二叉堆实现优先级队列
  7. 浅析ReDoS的原理与实践
  8. segyio库的使用
  9. Python之@property详解及底层实现介绍
  10. Java 将Excel转为et和ett格式