前言

本人在此题有一种不是题解的方法,但无法证明也找不到反例。

如果各位大神有反例或证明请发至

邮箱:qq1350742779@163.com

Description

Alice和Bob有一棵树(无根、无向),在第i个点上有ai个巧克力。首先,两人个选择一个起点(不同的),获得点上的巧克力;接着两人轮流操作(Alice先),操作的定义是:在树上找一个两人都没选过的点并获得点上的巧克力,并且这个点要与自己上一次选的点相邻。当有一人无法操作 时,另一个人可以继续操作,直到不能操作为止。因为Alice和Bob是好朋友,所以他们希望两人得到的巧克力总和尽量大,请输出最大总和。

Input

第一行一个整数n,表示树的点数

第二行有n个整数,表示每个点上的巧克力数量ai

接下来n-1行,每行两个整数u、v,表示u和v之间有一条边。

Output

输出一个整数,表示两人能获得得巧克力的最大总和。

Sample Input

9

1 2 3 4 5 6 7 8 9

1 2

1 3

1 4

1 5

1 6

1 7

1 8

1 9

Sample Output

25

Data Constraint

对于20%的数据,n<=15

对于40%的数据,n<=100

对于60%的数据,n<=5000

对于100%的数据,n<=200000,0<=ai<=1000000000(1e9)

我的方法

首先随便找一条直径,然后删掉这些的,在剩下的森林中再找一条最长的链,输出总和。

或者,

首先随便找一条直径,枚举这条直径上的两个点,从这两个点出发分别找到最长链,答案就是这条直径加上从枚举的两个点找到的最长链再减去枚举的两个点在直径之间的路径。

将两者去最大值输出。

#include <cmath>
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <queue>
const long long maxlongint=2147483647;
const long long mo=1000000007;
const long long N=200005;
using namespace std;
long long n,v[N],next[N*2],last[N*2],to[N*2],tot,ans,mxp,pos,bz[N],s,t,st[N],ts[N],num,ans1,ans2,ans3,ans4,ans5,vv;
long long bj(long long x,long long y)
{
next[++tot]=last[x];
last[x]=tot;
to[tot]=y;
}
long long dg(long long x,long long fa,long long sum)
{
if(sum>ans) ans=sum,mxp=x;
for(int i=last[x];i;i=next[i])
{
int j=to[i];
if(j!=fa && bz[j]==0) dg(j,x,sum+v[j]);
}
}
long long dg1(long long x,long long fa)
{
if(mxp==x && !bz[x]) bz[x]=vv;
for(int i=last[x];i;i=next[i])
{
int j=to[i];
if(j!=fa && bz[j]==0)
{
dg1(j,x);
if(bz[j])
{
bz[x]=vv;
break;
}
}
}
}
long long dg2(long long x,long long fa,long long sum)
{
ans=0;
dg(x,fa,0);
st[++tot]=ans+sum;
for(int i=last[x];i;i=next[i])
{
int j=to[i];
if(j!=fa && bz[j]==vv)
{
dg2(j,x,sum+v[j]);
}
}
}
long long dg3(long long x,long long fa,long long sum)
{
ans=0;
dg(x,fa,0);
ts[tot--]=ans+sum;
for(int i=last[x];i;i=next[i])
{
int j=to[i];
if(j!=fa && bz[j]==vv)
{
dg3(j,x,sum+v[j]);
}
}
}
int main()
{
scanf("%lld",&n);
for(long long i=1;i<=n;i++) scanf("%lld",&v[i]);
for(long long i=1;i<=n-1;i++)
{
long long x,y;
scanf("%lld%lld",&x,&y);
bj(x,y);
bj(y,x);
}
vv++;
dg(1,0,v[1]);
pos=mxp;
ans=0;
dg(pos,0,v[pos]);
dg1(pos,0);
s=mxp;
ans=0;
dg(pos,0,v[pos]);
dg1(pos,0);
t=mxp;
tot=0;
dg2(s,0,v[s]);
num=tot;
dg3(t,0,v[t]);
ans1=st[num];
for(int i=1;i<=num;i++) st[i]=max(st[i],st[i-1]);
for(int i=num;i>=1;i--) ts[i]=max(ts[i],ts[i+1]);
ans=0;
for(int i=1;i<=num-1;i++) ans=max(ans,st[i]+ts[i+1]);
ans5=ans;
for(int k=1;k<=n;k++)
if(!bz[k])
{
vv++;
tot=0;
num=0;
ans=0;
dg(k,0,v[k]);
pos=mxp;
ans=0;
dg(pos,0,v[pos]);
dg1(pos,0);
s=mxp;
ans=0;
dg(pos,0,v[pos]);
dg1(pos,0);
t=mxp;
tot=0;
dg2(s,0,v[s]);
num=tot;
ans3=max(ans3,st[num]);
}
printf("%lld",max(ans5,ans1+ans3));
}

最新文章

  1. 为jQuery添加Webkit的触摸方法支持
  2. Oracle中经典分页代码!
  3. PHP常量PHP_SAPI与函数php_sapi_name()简介,PHP运行环境检测
  4. 转-OpenJDK源码阅读导航跟编译
  5. 静态库不要strip 太厉害
  6. perl处理含有中文字符的json编码
  7. Eruda——手机网页前端调试面板
  8. RAW模板命名规范
  9. PHP缓存技术的多种方法小结
  10. [uva11992]Fast Matrix Operations(多延迟标记,二维线段树,区间更新)
  11. [刷题]算法竞赛入门经典(第2版) 4-6/UVa508 - Morse Mismatches
  12. Object-C中release的机制问题
  13. C编程之 一个容易忽视但是十分严重的小错误
  14. mycat入门_介绍与安装
  15. cookie和session 以及Django中应用
  16. JAVA基础部分复习(三、泛型)
  17. 使用Golang编写优化算法 (1)
  18. 通过Tacker将NFV引入OpenStack
  19. jQuery之双下拉框
  20. tty linux 打开和设置范例

热门文章

  1. Fidder插件自动生成爬虫代码(C#)
  2. 容易忽略的javascript知识点的总结
  3. Day05:访问控制 、 static和final
  4. JsonProperty 使用
  5. SQL子连接案例
  6. Myeclipse下配置SVN报错问题 svn: E175002: java.lang.RuntimeException: Could not generate DH keypair(转)
  7. for循环练习题:拆解字符并输入下标
  8. JSR303 校验扩展(分组、按顺序校验)
  9. 04: CI(持续集成)/CD(持续交付/持续部署)
  10. php前台表单限制PHP上传大小