51nod"省选"模测 A 树的双直径(树形dp)
题意
Sol
比赛结束后才调出来。。不多说啥了,就是因为自己菜。
裸的up-down dp,维护一下一个点上下的直径就行,一开始还想了个假的思路写了半天。。
转移都在代码注释里
毒瘤题目卡空间
#include<bits/stdc++.h>
#define Pair pair<int, int>
#define MP(x, y) make_pair(x, y)
#define fi first
#define se second
#define LL long long
using namespace std;
const int MAXN = 6e5 + 10;
template <typename A, typename B> inline bool chmin(A &a, B b){if(a > b) {a = b; return 1;} return 0;}
template <typename A, typename B> inline bool chmax(A &a, B b){if(a < b) {a = b; return 1;} return 0;}
template <typename A> inline void debug(A a){cout << a << '\n';}
template <typename A> inline LL sqr(A x){return 1ll * x * x;}
inline int read() {
char c = getchar(); int x = 0, f = 1;
while(c < '0' || c > '9') {if(c == '-') f = -1; c = getchar();}
while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return x * f;
}
void print(__int128 x) {
if (x>9) print(x/10);
putchar('0'+x%10);
}
int N, p[MAXN * 2], ve[MAXN * 2];
long long f[MAXN], g[MAXN], mxpre[MAXN], mxsuf[MAXN], dl[MAXN], ul[MAXN];
__int128 ans;
vector<Pair> v[MAXN];
void downdfs(int x, int fa) {
f[x] = dl[x] = 0;
for(auto &tp : v[x]) {
int to = tp.fi, w = tp.se;
if(to == fa) continue;
downdfs(to, x);
chmax(dl[x], f[x] + f[to] + w);
chmax(f[x], f[to] + w);
chmax(dl[x], dl[to]);
}
}
void updfs(int x, int fa) {
int cnt = 0;
for(int i = 0; i < v[x].size(); i++)
if(v[x][i].fi != fa) p[++cnt] = v[x][i].fi, ve[cnt] = v[x][i].se;
mxpre[0] = 0; mxsuf[cnt + 1] = 0;
for(int i = 1; i <= cnt; i++) mxpre[i] = max(mxpre[i - 1], f[p[i]] + ve[i]);
for(int i = cnt; i >= 1; i--) mxsuf[i] = max(mxsuf[i + 1], f[p[i]] + ve[i]);
//一个点向上的直径:
//1: 从父亲的ul继承过来
//2: 前后缀中的最大值f + 出边 + 入边
//3: 父亲的g + 兄弟节点中最大的f + 出边
//4: 前驱/后继 中的最大和次大
//5: 前驱/后继 中的子树中的直径
for(int i = 1; i <= cnt; i++) {
int to = p[i], w = ve[i];
chmax(g[to], g[x] + w);
chmax(g[to], max(mxpre[i - 1], mxsuf[i + 1]) + w);
chmax(ul[to], ul[x]);//1
chmax(ul[to], mxpre[i - 1] + mxsuf[i + 1]);//2
chmax(ul[to], g[x] + max(mxpre[i - 1], mxsuf[i + 1]));//3
}
long long mx1 = -1e18, mx2 = -1e18, mx3 = -1e18;
for(int i = 1; i <= cnt; i++) {
chmax(ul[p[i]], max(mx1 + mx2, mx3)); int tmp = f[p[i]] + ve[i];
if(tmp > mx1) chmax(mx2, mx1), mx1 = tmp;
else if(tmp > mx2) mx2 = tmp;
chmax(mx3, dl[p[i]]);
}
mx1 = -1e18, mx2 = -1e18, mx3 = -1e18;
for(int i = cnt; i >= 1; i--) {
chmax(ul[p[i]], max(mx1 + mx2, mx3)); int tmp = f[p[i]] + ve[i];
if(tmp > mx1) chmax(mx2, mx1), mx1 = tmp;
else if(tmp > mx2) mx2 = tmp;
chmax(mx3, dl[p[i]]);
}
for(int i = 0; i < v[x].size(); i++)
if(v[x][i].fi != fa) updfs(v[x][i].fi, x);
}
signed main() {
//freopen("a.in", "r", stdin);
N = read();
for(int i = 1; i <= N - 1; i++) {
int x = read(), y = read(), w = read();
v[x].push_back(MP(y, w));
v[y].push_back(MP(x, w));
}
downdfs(1, 0);
updfs(1, 0);
for(int i = 2; i <= N; i++) chmax(ans, (__int128) dl[i] * ul[i]);
for(int i = 1; i <= N; i++) for(auto &tp : v[i]) tp.se = -tp.se;
memset(ul, 0, sizeof(ul));
memset(g, 0, sizeof(g));
downdfs(1, 0);
updfs(1, 0);
for(int i = 2; i <= N; i++) chmax(ans, (__int128) dl[i] * ul[i]);
print(ans);
//cout << ans;
return 0;
}
51nod"省选"模测 A 树的双直径(树形dp)的更多相关文章
- 51nod“省选”模测第二场 C 小朋友的笑话(线段树 set)
题意 题目链接 Sol 直接拿set维护\(li\)连续段.因为set内的区间互不相交,而且每个线段会被至多加入删除一次,所以复杂度是对的. #include<bits/stdc++.h> ...
- 51nod“省选”模测第二场 B 异或约数和(数论分块)
题意 题目链接 Sol 这题是来搞笑的吧.. 考虑一个数的贡献是\(O(\frac{N}{i})\) 直接数论分块. #include<bits/stdc++.h> #define Pai ...
- bzoj 3302&2447&2103 树的双中心 树形DP
题目: 题解: bzoj 3302 == 2447 == 2103 三倍经验 首先我们考虑枚举两个中心的位置,然后统计答案. 我们发现,一定有一部分点离第一个中心更近,另一部分点离第二个中心更近 如果 ...
- 51nod 1812 树的双直径 题解【树形DP】【贪心】
老了-稍微麻烦一点的树形DP都想不到了. 题目描述 给定一棵树,边权是整数 \(c_i\) ,找出两条不相交的链(没有公共点),使得链长的乘积最大(链长定义为这条链上所有边的权值之和,如果这条链只有 ...
- [CEOI2007]树的匹配Treasury(树形DP+高精)
题意 给一棵树,你可以匹配有边相连的两个点,问你这棵树的最大匹配时多少,并且计算出有多少种最大匹配. N≤1000,其中40%的数据答案不超过 108 题解 显然的树形DP+高精. 这题是作为考试题考 ...
- Codeforces 633F 树的直径/树形DP
题意:有两个小孩玩游戏,每个小孩可以选择一个起始点,并且下一个选择的点必须和自己选择的上一个点相邻,问两个选的点权和的最大值是多少? 思路:首先这个问题可以转化为求树上两不相交路径的点权和的最大值,对 ...
- 2014 Super Training #9 E Destroy --树的直径+树形DP
原题: ZOJ 3684 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3684 题意: 给你一棵树,树的根是树的中心(到其 ...
- 算法笔记--树的直径 && 树形dp && 虚树 && 树分治 && 树上差分 && 树链剖分
树的直径: 利用了树的直径的一个性质:距某个点最远的叶子节点一定是树的某一条直径的端点. 先从任意一顶点a出发,bfs找到离它最远的一个叶子顶点b,然后再从b出发bfs找到离b最远的顶点c,那么b和c ...
- BZOJ5123 线段树的匹配(树形dp)
线段树的任意一棵子树都相当于节点数与该子树相同的线段树.于是假装在树形dp即可,记忆化搜索实现,有效状态数是logn级别的. #include<iostream> #include< ...
随机推荐
- [Leetcode]895.最大频率栈
Problem 实现 FreqStack,模拟类似栈的数据结构的操作的一个类. FreqStack 有两个函数: push(int x),将整数 x 推入栈中. pop(),它移除并返回栈中出现最频繁 ...
- murri
github: https://github.com/haltu/muuri 官网:https://haltu.github.io/muuri/ 安装 npm install murri —sav ...
- 总结jenkins Android自动打包遇到的坑
一.ndk-build报错 [root@hejianlai-jenkins LearnGradle]# ndk-build /usr/local/android-ndk-r8/ndk-build: / ...
- 安装mono和monoDevelop开发环境
我之前用的是Ubuntu10.12版本的linux,一直被软件更新包困扰,总是保存缺少依赖包的问题!总之在网络上找到的软件更新包地址都不能解决问题.后来就用了比较新的TLS版本Ubuntu14.04. ...
- 微信分享config:ok 但自定义内容无效
一.问题 使用微信 JSSDK 分享,出现自定义内容无效 ,也就是分享出去的内容不是你配置的内容. 但在调试过程中发现 congfig 都是 ok 的 二.解决 检查config 配置是否正确 js ...
- 项目ITP(四) javaweb http json 交互 in action (服务端 spring 手机端 提供各种工具类)勿喷!
前言 系列文章:[传送门] 洗了个澡,准备写篇博客.然后看书了.时间 3 7 分.我慢慢规律生活,向目标靠近. 很喜欢珍惜时间像叮当猫一样 正文 慢慢地,二维码实现签到将要落幕了.下篇文章出二维码实 ...
- TCP/IP 笔记 - TCP拥塞控制
拥塞控制是TCP通信的每一方需要执行的一系列行为,这些行为有特定算法规定,用于防止网络因为大规模的通信负载而瘫痪.其基本方法是当有理由认为网络即将进入拥塞状态(或已由于拥塞而出现路由丢包情况)时减缓T ...
- Mycat - 实现数据库的读写分离与高可用
前言 开心一刻 上语文课,不小心睡着了,坐在边上的同桌突然叫醒了我,并小声说道:“读课文第三段”.我立马起身大声读了起来.正在黑板写字的老师吓了一跳,老师郁闷的看着我,问道:“同学有什么问题吗?”,我 ...
- SpringBoot之打成war包部署到Tomcat
正常情况下SpringBoot项目是以jar包的形式,正常情况下SpringBoot项目是以jar包的形式,并且SpringBoot是内嵌Tomcat服务器,所以每次重新启动都是用的新的Tomcat服 ...
- Python系列:五、异常处理-技术流ken
至今为止还没有进一步的谈论过错误信息,不过在你已经试验过的那些例子中,可能已经遇到过一些.Python 中(至少)有两种错误:语法错误和异常( syntax errors 和 exceptions ) ...