思路:我们其实只需要枚举每条边,求得最小值就行了。

先dfs算出每个节点作为根时,其子树的最长路径,以及进过该节点的最长,次长,第三长路径。

然后在次dfs枚举求出切断某条边,求出这条边的两个端点为子树根,其子树的最长路径。b就是其中较大的。

#pragma comment(linker, "/STACK:1024000000,1024000000")
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<cstdio>
#include<vector>
#define Maxn 200010
using namespace std;
int road[Maxn],Max[Maxn],lMax[Maxn],vi[Maxn],e,head[Maxn],dis[Maxn],ldis[Maxn],pr,ans,sMax[Maxn],lroad[Maxn];
struct Edge{
int u,v,val,id,next;
}edge[Maxn*];
void init()
{
memset(road,,sizeof(road));
memset(Max,,sizeof(Max));
memset(lMax,,sizeof(lMax));
memset(vi,,sizeof(vi));
memset(dis,,sizeof(dis));
memset(ldis,,sizeof(ldis));
memset(head,-,sizeof(head));
memset(sMax,,sizeof(sMax));
memset(lroad,,sizeof(lroad));
e=;
}
void add(int u,int v,int val,int id)
{
edge[e].u=u,edge[e].v=v,edge[e].val=val,edge[e].id=id,edge[e].next=head[u],head[u]=e++;
edge[e].u=v,edge[e].v=u,edge[e].val=val,edge[e].id=id,edge[e].next=head[v],head[v]=e++;
}
void dfs(int u)
{
int i,v;
vi[u]=;
for(i=head[u];i!=-;i=edge[i].next)
{
v=edge[i].v;
if(vi[v]) continue;
dfs(v);
if(Max[v]+>Max[u])
{
sMax[u]=lMax[u];
lroad[u]=road[u];
lMax[u]=Max[u];
Max[u]=Max[v]+;
road[u]=v;
}
else
if(Max[v]+>lMax[u])
{
sMax[u]=lMax[u];
lMax[u]=Max[v]+;
lroad[u]=v;
}
dis[u]=max(dis[u],Max[u]+lMax[u]);
dis[u]=max(dis[u],dis[v]);
}
}
void predfs(int u,int d,int pre,int fa)
{
int i,v;
vi[u]=;
int now=;
int a,b=;
for(i=head[u];i!=-;i=edge[i].next)
{
v=edge[i].v;
if(vi[v]) continue;
a=edge[i].val;
now=pre;
if(road[u]==v)
{
now=max(d+lMax[u],now);
now=max(lMax[u]+sMax[u],now);
b=max(now,dis[v]);
if(a*b<=pr)
{
if(a*b==pr)
{
if(edge[i].id<ans)
ans=edge[i].id;
}
else
pr=a*b,ans=edge[i].id;
}
predfs(v,max(d,lMax[u])+,now,u);
}
else
if(lroad[u]==v)
{
now=max(d+Max[u],now);
now=max(Max[u]+sMax[u],now);
b=max(now,dis[v]);
if(a*b<=pr)
{
if(a*b==pr)
{
if(edge[i].id<ans)
ans=edge[i].id;
}
else
pr=a*b,ans=edge[i].id;
}
predfs(v,max(d,Max[u])+,now,u);
}
else
{
now=max(d+Max[u],now);
now=max(Max[u]+lMax[u],now);
b=max(now,dis[v]);
if(a*b<=pr)
{
if(a*b==pr)
{
if(edge[i].id<ans)
ans=edge[i].id;
}
else
pr=a*b,ans=edge[i].id;
}
predfs(v,max(d,Max[u])+,now,u);
}
}
}
int main()
{
int n,i,j,m,t,u,v,w,Case=;
scanf("%d",&t);
while(t--)
{
init();
scanf("%d",&n);
for(i=;i<=n-;i++)
{
scanf("%d%d%d",&u,&v,&w);
add(u,v,w,i);
}
int a,b;
dfs();
memset(vi,,sizeof(vi));
pr=;
ans=;
predfs(,,,);
printf("Case #%d: %d\n",++Case,ans);
}
return ;
}

hdu 4679 树状dp的更多相关文章

  1. hdu 4582 树状DP

    思路:首先声明我是参考:http://blog.csdn.net/frog1902/article/details/9921845这位大牛的博客的. 他说的已经很详尽,但我还是要补充几点. 看完他的解 ...

  2. hdu 4340 树状DP

    思路:我们定义两个数组,ant[Maxn][2],bob[Maxn][2].ant[i][0]表示还未确定哪个城市被全费用占领,ant[i][1]表示确定了哪个城市被全费用占领.那么ant[i][0] ...

  3. HDU 4035Maze(树状+概率dp,绝对经典)

    题意: 给你n个节点的树,从1节点开始走,到每个节点都有三种情况,被杀死回到1节点,找到隐藏的出口出去,沿着当前节点相邻的边走到下一个节点,给出每个节点三种情况发生的概率分别为ki,ei,1-ki-e ...

  4. hdu 1561 The more, The Better_树状dp

    题目链接 题意:给你一棵树,各个节点都有价值(除根节点),从根节点出发,选择m个节点,问最多的价值是多小. 思路:很明显是树状dp,遍历树时背包最优价值,dp[i][k]=max{dp[i][r]+d ...

  5. HDU 4714 Tree2cycle(树状DP)(2013 ACM/ICPC Asia Regional Online ―― Warmup)

    Description A tree with N nodes and N-1 edges is given. To connect or disconnect one edge, we need 1 ...

  6. 树状DP HDU1520 Anniversary party

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1520 题意:职员之间有上下级关系,每个职员有自己的happy值,越高在派对上就越能炒热气氛.但是必须是 ...

  7. 树状DP (poj 2342)

    题目:Anniversary party 题意:给出N各节点的快乐指数,以及父子关系,求最大快乐指数和(没人职员愿意跟直接上司一起玩): 思路:从底向上的树状DP: 第一种情况:第i个员工不参与,F[ ...

  8. poj3659树状DP

    Cell Phone Network Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 6273   Accepted: 225 ...

  9. poj 2342 Anniversary party_经典树状dp

    题意:Ural大学有n个职员,1~N编号,他们有从属关系,就是说他们关系就像一棵树,父节点就是子节点的直接上司,每个职员有一个快乐指数,现在要开会,职员和职员的直接上司不能同时开会,问怎才能使开会的快 ...

随机推荐

  1. JDBC学习笔记(2)——Statement和ResultSet

    Statement执行更新操作 Statement:Statement 是 Java 执行数据库操作的一个重要方法,用于在已经建立数据库连接的基础上,向数据库发送要执行的SQL语句.Statement ...

  2. POJ3126 Prime Path

    http://poj.org/problem?id=3126 题目大意:给两个数四位数m, n, m的位数各个位改变一位0 —— 9使得改变后的数为素数, 问经过多少次变化使其等于n 如: 10331 ...

  3. codeforces 630 I(规律&&组合)

    I - Parking Lot Time Limit:500MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64u Subm ...

  4. HTTP协议状态码详解

    HTTP状态码,我都是现查现用. 我以前记得几个常用的状态码,比如200,302,304,404, 503. 一般来说我也只需要了解这些常用的状态码就可以了.  如果是做AJAX,REST,网络爬虫, ...

  5. 如何让自己的电脑发布ASP http://jingyan.baidu.com/article/19192ad853224ce53f570748.html

    怎样在WIN7系统下安装IIS | 浏览:122821 | 更新:2012-03-03 14:07 | 标签:windows7 1 2 3 4 5 6 7 分步阅读 在此根据多年的网站开发经验,把如何 ...

  6. 加粗合并latex表格线的加粗及合并两行

    每日一贴,今天的内容关键字为加粗合并 在latex中要设置加粗的表格线,要使用如下包: \usepackage{booktabs} 如下图中的表格,首行(\toprule[2pt]),旁边行(\mid ...

  7. SAP BW顾问如何保持市场竞争力

    跟大部分电工一样,SAP顾问也经常有迷茫的时候.因为,这个世界变化实在太快了.每一个电工,总是在担心自己会不会被飞速发展的技术所淘汰.那么,作为 一个BW顾问,应该如何保持市场竞争力呢?我觉得需要两个 ...

  8. ORACLE 11G用于有效期

    Oracle报错,ORA-28001: 口令已经失效(转自网络) 错误信息:ORA-28001: the password has expired解决方法 Oracle11G创建用户时缺省passwo ...

  9. 返回ListBox选中的多项目

    //返回ListBox选中的多项目 procedure TForm1.Button2Click(Sender: TObject);vari:Integer;s:string;begin   for i ...

  10. Python2.7.3移除字符串中重复字符(一)

    移除重复字符很简单,这里是最笨,也是最简单的一种.问题关键是理解排序的意义: # coding=utf-8 #learning at jeapedu in 2013/10/26 #移除给定字符串中重复 ...