SCUT - 274 - CC B-Tree - 树形dp
首先要判断是一颗树,并且找出树的直径。
是一棵树,首先边恰好有n-1条,其次要连通,这两个条件已经充分了,当然判环可以加速。
两次dfs找出直径,一边叫做L,另一边叫做R。(第一次写这个)
然后树形dp。
规定其中一个叶子作为树根。然后fx表示从x向下(叶子)走能走到的最远距离,这个非常简单。
然后漏了什么情况呢?从x向上走的情况。
这个时候要从根开始维护一个叫做gx的数组,那么每次孩子v的gx就是父亲u的gx(继续向上走)和u的fx(从父亲开始往下走)的最大值。注意这个时候往下走的不能走回去v,所以要记录两个fx,并且记录大的那个是走哪个方向的。
边界:在叶子/树根等不能走的位置,根据定义,距离为0,点序号为本身。
#include<bits/stdc++.h>
using namespace std;
const int MAXN = 1e5 + 5;
const int INF = 0x3f3f3f3f;
struct Edge {
int v, w;
};
vector<Edge> e[MAXN];
void init0(int n) {
for(int i = 0; i <= n; ++i)
e[i].clear();
}
int cnt, vis[MAXN], dis[MAXN], maxdis, L, R;
void init1(int n) {
memset(vis, false, sizeof(vis[0]) * (n + 1));
memset(dis, 0, sizeof(dis[0]) * (n + 1));
cnt = 0, maxdis = 0;
}
bool dfs1(int id, int fa) {
vis[id] = true, ++cnt;
for(auto p : e[id]) {
int v = p.v, w = p.w;
if(v == fa)
continue;
if(vis[v])
return false;
dis[v] = dis[id] + w;
if(dis[v] > maxdis)
maxdis = dis[v], L = v;
if(dfs1(v, id) == false)
return false;
}
return true;
}
void init2(int n) {
memset(dis, 0, sizeof(dis[0]) * (n + 1));
maxdis = 0;
}
void dfs2(int id, int fa) {
for(auto p : e[id]) {
int v = p.v, w = p.w;
if(v == fa)
continue;
dis[v] = dis[id] + w;
if(dis[v] > maxdis)
maxdis = dis[v], R = v;
dfs2(v, id);
}
}
struct F {
int v, w, s;
F(int v = 0, int w = 0, int s = 0): v(v), w(w), s(s) {}
} f[MAXN], f2[MAXN], tmp;
void maintainF(int id) {
if(tmp.w > f2[id].w || tmp.w == f2[id].w && tmp.v < f2[id].v) {
f2[id] = tmp;
if(f2[id].w > f[id].w || f2[id].w == f[id].w && f2[id].v < f[id].v)
swap(f[id], f2[id]);
}
}
void dfs3(int id, int fa) {
f[id] = f2[id] = F(id, 0, -1);
for(auto p : e[id]) {
int v = p.v, w = p.w;
if(v == fa)
continue;
dfs3(v, id);
tmp.w = f[v].w + w, tmp.v = f[v].v, tmp.s = v;
maintainF(id);
}
}
struct G {
int v, w;
} g[MAXN];
F getF(int id, int fa) {
if(f[fa].s == id)
return f2[fa];
return f[fa];
}
void maintainG(int id, int fa, int faw) {
if(fa == 0) {
g[id].w = 0;
g[id].v = id;
return;
}
tmp = getF(id, fa);
if(g[fa].w > tmp.w) {
g[id].w = g[fa].w + faw;
g[id].v = g[fa].v;
} else if(g[fa].w == tmp.w) {
g[id].w = g[fa].w + faw;
g[id].v = min(tmp.v, g[fa].v);
} else {
g[id].w = tmp.w + faw;
g[id].v = tmp.v;
}
}
void dfs4(int id, int fa, int faw) {
maintainG(id, fa, faw);
for(auto p : e[id]) {
int v = p.v, w = p.w;
if(v == fa)
continue;
dfs4(v, id, w);
}
}
int getFG(int id) {
if(f[id].w > g[id].w)
return f[id].v;
else if(f[id].w == g[id].w)
return min(f[id].v, g[id].v);
return g[id].v;
}
int main() {
#ifdef Yinku
freopen("Yinku.in", "r", stdin);
#endif // Yinku
int n, m, k;
while(~scanf("%d%d%d", &n, &m, &k)) {
if(m != n - 1) {
for(int i = 1, u, v, w; i <= m; ++i)
scanf("%d%d%d", &u, &v, &w);
puts("There is no B-Tree");
continue;
}
init0(n);
for(int i = 1, u, v, w; i <= m; ++i) {
scanf("%d%d%d", &u, &v, &w);
e[u].push_back({v, w});
e[v].push_back({u, w});
}
init1(n);
if(dfs1(1, 0) == false || cnt != n) {
puts("There is no B-Tree");
continue;
}
init2(n);
dfs2(L, 0);
if(dis[R] > k) {
puts("There is no B-Tree");
continue;
}
dfs3(R, 0);
dfs4(R, 0, 0);
for(int i = 1; i <= n; ++i)
printf("%d\n", getFG(i));
}
}
SCUT - 274 - CC B-Tree - 树形dp的更多相关文章
- 熟练剖分(tree) 树形DP
熟练剖分(tree) 树形DP 题目描述 题目传送门 分析 我们设\(f[i][j]\)为以\(i\)为根节点的子树中最坏时间复杂度小于等于\(j\)的概率 设\(g[i][j]\)为当前扫到的以\( ...
- hdu-5834 Magic boy Bi Luo with his excited tree(树形dp)
题目链接: Magic boy Bi Luo with his excited tree Time Limit: 8000/4000 MS (Java/Others) Memory Limit: ...
- CF 461B Appleman and Tree 树形DP
Appleman has a tree with n vertices. Some of the vertices (at least one) are colored black and other ...
- codeforces 161D Distance in Tree 树形dp
题目链接: http://codeforces.com/contest/161/problem/D D. Distance in Tree time limit per test 3 secondsm ...
- hdu6035 Colorful Tree 树形dp 给定一棵树,每个节点有一个颜色值。定义每条路径的值为经过的节点的不同颜色数。求所有路径的值和。
/** 题目:hdu6035 Colorful Tree 链接:http://acm.hdu.edu.cn/showproblem.php?pid=6035 题意:给定一棵树,每个节点有一个颜色值.定 ...
- 5.10 省选模拟赛 tree 树形dp 逆元
LINK:tree 整场比赛看起来最不可做 确是最简单的题目. 感觉很难写 不过单独考虑某个点 容易想到树形dp的状态. 设f[x]表示以x为根的子树内有黑边的方案数. 白边方案只有一种所以不用记录. ...
- Codeforces Round #263 Div.1 B Appleman and Tree --树形DP【转】
题意:给了一棵树以及每个节点的颜色,1代表黑,0代表白,求将这棵树拆成k棵树,使得每棵树恰好有一个黑色节点的方法数 解法:树形DP问题.定义: dp[u][0]表示以u为根的子树对父亲的贡献为0 dp ...
- codeforces Round #263(div2) D. Appleman and Tree 树形dp
题意: 给出一棵树,每个节点都被标记了黑或白色,要求把这棵树的其中k条变切换,划分成k+1棵子树,每颗子树必须有1个黑色节点,求有多少种划分方法. 题解: 树形dp dp[x][0]表示是以x为根的树 ...
- POJ 2486 Apple Tree(树形DP)
题目链接 树形DP很弱啊,开始看题,觉得貌似挺简单的,然后发现貌似还可以往回走...然后就不知道怎么做了... 看看了题解http://www.cnblogs.com/wuyiqi/archive/2 ...
随机推荐
- [REPRINT]Properties vs. Getters and Setters
http://www.python-course.eu/python3_properties.php Our new class means breaking the interface. The a ...
- React Native 之createDrawerNavigator和createSwitchNavigator
其他代码接上篇文章 createDrawerNavigator 抽屉 createSwitchNavigator 模拟登录=>主界面 index.js /** * @format */ impo ...
- HDU 6230 Palindrome ( Manacher && 树状数组)
题意 : 给定一个字符串S,问你有多少长度为 n 的子串满足 S[i]=S[2n−i]=S[2n+i−2] (1≤i≤n) 参考自 ==> 博客 分析 : 可以看出满足题目要求的特殊回文子串其 ...
- web classpath路径说明
http://blog.csdn.net/wyswlp/article/details/8886385 http://www.cnblogs.com/Ant-soldier/p/5474085.htm ...
- open 函数处理文件
open函数用于文件处理 操作文件时,一般需要经历如下步骤:1 打开文件 2 操作文件 f = open("文件名" , ' 打开文件方式' ) 文件句柄 ...
- jetty 启动时出现的问题
启动时出现: 1. NoClassDefFoundError: javax/xml/registry/infomodel/User 是需要导入jaxr-api包, <dependency> ...
- What is the difference between Kill and Kill -9 command in Unix?
w difference kill -9 pid and kill pid command - Ask Ubuntu https://askubuntu.com/questions/791841/d ...
- Python字典的json格式化处理(换行与不换行)
Prefer = {"jim": {"War": 1.9, "the big bang": 1.0, "The lord of w ...
- Android studio 不能创建Activity等文件
这是我之前安装Android studio的一系列问题:http://tieba.baidu.com/p/5921373177 1. 不能创建Activity等许多文件: 2. 工程运行不了: 3. ...
- 阶段1 语言基础+高级_1-3-Java语言高级_1-常用API_1_第6节 static静态_11_静态static关键字概述
static静态关键字的基本含义 红色教室都划掉 一份数据分给三个人用,而不是三个人相同的数据,三个对象写三遍没有必要. 只写一遍教室,写在学生的类当中.这样还就可以省内存.如果要换教室的话 ,只需要 ...