Manthan, Codefest 16 F
寻找树上最大权值和的两条不相交的路径。
树形DP题。挺难的,对于我……
定义三个变量ma[MAXN], t[MAXN], sum[MAXN]
其中,ma[i]代表i子树中,最长的路径和
t[i]代表i子树中,用来维护已有一条路径,而且还有一条链从叶子节点到i,则可以从根节点i向上扩展。如下图,维护红色部分

sum[i]维护从某叶子节点到根节点i的最长路径。
转移方程可以看代码,很容易明白
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <cmath>
#include <set>
#define LL long long using namespace std; const int MAXN = 100050;
const int MOD = 1e9 + 7; LL ans;
LL ma[MAXN], t[MAXN], sum[MAXN];
LL l[MAXN], ml[MAXN], r[MAXN], mr[MAXN];
bool leaf[MAXN];
LL mm[10];
bool vis[MAXN];
int head[MAXN];
struct Edge{
int u, v;
int next;
}edge[MAXN * 2];
int weight[MAXN], par[MAXN];
int n, tot; void addedge(int u, int v){
edge[tot].u = u;
edge[tot].v = v;
edge[tot].next = head[u];
head[u] = tot++;
} void dfs(int u){
vis[u] = true;
leaf[u] = true;
for(int e = head[u]; e != -1; e = edge[e].next){
int v = edge[e].v;
if(vis[v]) continue;
par[v] = u;
dfs(v);
leaf[u] = false;
}
} void slove(int u){
if(leaf[u]){
ma[u] = t[u] = sum[u] = weight[u];
return ;
} LL m1, m2, M1, M2;
m1 = m2 = M1 = M2 = 0;
for(int e = head[u]; e != -1; e = edge[e].next){
int v = edge[e].v;
if(v != par[u]){
slove(v);
if(ma[v] >= M1){
M2 = M1, M1 = ma[v];
}
else if(ma[v] > M2){
M2 = ma[v];
}
if(sum[v] >= m1){
m2 = m1, m1 = sum[v];
}
else if(sum[v] > m2){
m2 = sum[v];
}
t[u] = max(t[u], t[v] + weight[u]);
}
}
ma[u] = max(M1, m1 + m2 + weight[u]);
sum[u] = m1 + weight[u];
ans = max(ans, M1 + M2); int counts = 0;
for(int e = head[u]; e != -1; e = edge[e].next){
int v = edge[e].v;
if(v != par[u]){
l[++counts] = sum[v];
r[counts] = sum[v];
}
}
l[0] = ml[0] = r[counts + 1] = mr[counts + 1] = 0; //从左往右寻找最大的两个sum for(int i = 1; i <= counts ; i++){
if(l[i] > l[i - 1]) ml[i] = l[i - 1];
else if(l[i] > ml[i - 1]){
ml[i] = l[i];
l[i] = l[i - 1];
}
else{
l[i] = l[i - 1], ml[i] = ml[i - 1];
}
} //从右往左。。。。 for(int i = counts; i >= 1; i--){
if(r[i] > r[i + 1]) mr[i] = r[i + 1];
else if(r[i] > mr[i + 1]){
mr[i] = r[i];
r[i] = r[i + 1];
}
else{
r[i] = r[i + 1], mr[i] = mr[i + 1];
}
} counts = 0;
for(int e = head[u]; e != -1; e = edge[e].next){
int v = edge[e].v;
if(v == par[u]) continue;
counts ++;
mm[0] = l[counts - 1], mm[1] = ml[counts - 1];
mm[2] = r[counts + 1], mm[3] = mr[counts + 1]; sort(mm, mm + 4); ans = max(ans, weight[u] + ma[v] + mm[3] + mm[2]);
ans = max(ans, weight[u] + mm[3] + t[v]);
t[u] = max(t[u], ma[v] + weight[u] + mm[3]);
} } int main(){
scanf("%d", &n);
memset(head, -1, sizeof(head));
// memset(vis, false, sizeof(vis));
// memset(leaf, false, sizeof(leaf));
tot = 0;
memset(t, 0, sizeof(t));
for(int i = 1; i <= n; i++){
scanf("%d", &weight[i]);
}
int u, v;
memset(par, -1, sizeof(par));
for(int i = 0; i < n - 1; i++){
scanf("%d%d", &u, &v);
addedge(u, v);
addedge(v, u);
}
dfs(1);
ans = 0;
slove(1); cout << ans << endl;
}
Manthan, Codefest 16 F的更多相关文章
- Manthan, Codefest 16
暴力 A - Ebony and Ivory import java.util.*; import java.io.*; public class Main { public static void ...
- CF Manthan, Codefest 16 G. Yash And Trees 线段树+bitset
题目链接:http://codeforces.com/problemset/problem/633/G 大意是一棵树两种操作,第一种是某一节点子树所有值+v,第二种问子树中节点模m出现了多少种m以内的 ...
- Manthan, Codefest 16 H. Fibonacci-ish II 大力出奇迹 莫队 线段树 矩阵
H. Fibonacci-ish II 题目连接: http://codeforces.com/contest/633/problem/H Description Yash is finally ti ...
- Manthan, Codefest 16 C. Spy Syndrome 2 字典树 + dp
C. Spy Syndrome 2 题目连接: http://www.codeforces.com/contest/633/problem/C Description After observing ...
- Manthan, Codefest 16 C
建trie树,刚好字符串是反向的,直接在原图上向前搜索就OK了……………… 可怜的我竟然用了RK来hash,在test67那里T了…… 贴个RK的 #include <iostream> ...
- Manthan, Codefest 16 D. Fibonacci-ish
D. Fibonacci-ish time limit per test 3 seconds memory limit per test 512 megabytes input standard in ...
- Manthan, Codefest 16(B--A Trivial Problem)
B. A Trivial Problem time limit per test 2 seconds memory limit per test 256 megabytes input standar ...
- Manthan, Codefest 16 -C. Spy Syndrome 2
time limit per test 2 seconds memory limit per test 256 megabytes input standard input output standa ...
- Manthan, Codefest 16 -A Ebony and Ivory
time limit per test 2 seconds memory limit per test 256 megabytes input standard input output standa ...
随机推荐
- WCF学习笔记(1)-一个完整的例子
一.开发环境 IDE:VS2013 OS:Win10 IIS:IIS 10 二.开发流程 1.项目结构 2.添加一个WCF程序 3.删除系统自动生成的两个文件IService1.cs和Service1 ...
- P1044 栈
题目背景 栈是计算机中经典的数据结构,简单的说,栈就是限制在一端进行插入删除操作的线性表. 栈有两种最重要的操作,即pop(从栈顶弹出一个元素)和push(将一个元素进栈). 栈的重要性不言自明,任何 ...
- Visual Studio 2017 无法连接到Web服务器"IIS Express"
.net core2.2 无法连接到Web服务器"IIS Express" 解决方案: 用命令提示符输入以下命令 sc config http start= auto 重启计算机, ...
- 在WEB项目中调用QQ通讯组件打开QQ聊天界面
在很多WEB项目中,需要提供在线服务的功能,加上自己的联系方式,例如:QQ,不用添加QQ好友也可以交谈,那这到底是怎么实现的呢? 对于这个功能,需要提到一个组件,即“QQ通讯组件”.QQ通讯组件是一种 ...
- 数据库——DBUtils和连接池
第一章 DBUtils如果只使用JDBC进行开发,我们会发现冗余代码过多,为了简化JDBC开发,本案例我们讲采用apache commons组件一个成员:DBUtils.DBUtils就是JDBC的简 ...
- checkbox prop无效问题
因为bootstrap插件问题,需要先获取input的上级元素,然后添加checked $("input[name='checkInput']").parent().addClas ...
- Qt 如何处理密集型耗时的事情
有时候需要处理一些跟界面无关的但非常耗时的事情,这些事情跟界面在同一个线程中,由于时间太长,导致界面无法响应,处于“假死”状态.例如:在应用程序中保存文件到硬盘上,从开始保存直到文件保存完毕,程序不响 ...
- 反片语(Ananagrams,Uva 156)
输入一些单词,找出所有满足如下条件的单词:该单词不能通过字母重排,得到输入文 本中的另外一个单词.在判断是否满足条件时,字母不分大小写,但在输出时应保留输入中 的大小写,按字典序进行排列(所有大写字母 ...
- Python使用Flask框架,结合Highchart处理csv数据(引申-从文件获取数据--从数据库获取数据)
参考链接:https://www.highcharts.com.cn/docs/process-text-data-file 1.javascript代码 var options = { chart: ...
- 作业 3-5 switch语句的应用
/*输入五级制成绩(A-E),输出相应的百分制成绩(0-100)区间*/ #include<stdio.h> int main(void) { char ch;/*定义一个字符*/ pri ...