https://scut.online/p/274

首先要判断是一颗树,并且找出树的直径。

是一棵树,首先边恰好有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的更多相关文章

  1. 熟练剖分(tree) 树形DP

    熟练剖分(tree) 树形DP 题目描述 题目传送门 分析 我们设\(f[i][j]\)为以\(i\)为根节点的子树中最坏时间复杂度小于等于\(j\)的概率 设\(g[i][j]\)为当前扫到的以\( ...

  2. 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: ...

  3. 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 ...

  4. codeforces 161D Distance in Tree 树形dp

    题目链接: http://codeforces.com/contest/161/problem/D D. Distance in Tree time limit per test 3 secondsm ...

  5. hdu6035 Colorful Tree 树形dp 给定一棵树,每个节点有一个颜色值。定义每条路径的值为经过的节点的不同颜色数。求所有路径的值和。

    /** 题目:hdu6035 Colorful Tree 链接:http://acm.hdu.edu.cn/showproblem.php?pid=6035 题意:给定一棵树,每个节点有一个颜色值.定 ...

  6. 5.10 省选模拟赛 tree 树形dp 逆元

    LINK:tree 整场比赛看起来最不可做 确是最简单的题目. 感觉很难写 不过单独考虑某个点 容易想到树形dp的状态. 设f[x]表示以x为根的子树内有黑边的方案数. 白边方案只有一种所以不用记录. ...

  7. Codeforces Round #263 Div.1 B Appleman and Tree --树形DP【转】

    题意:给了一棵树以及每个节点的颜色,1代表黑,0代表白,求将这棵树拆成k棵树,使得每棵树恰好有一个黑色节点的方法数 解法:树形DP问题.定义: dp[u][0]表示以u为根的子树对父亲的贡献为0 dp ...

  8. codeforces Round #263(div2) D. Appleman and Tree 树形dp

    题意: 给出一棵树,每个节点都被标记了黑或白色,要求把这棵树的其中k条变切换,划分成k+1棵子树,每颗子树必须有1个黑色节点,求有多少种划分方法. 题解: 树形dp dp[x][0]表示是以x为根的树 ...

  9. POJ 2486 Apple Tree(树形DP)

    题目链接 树形DP很弱啊,开始看题,觉得貌似挺简单的,然后发现貌似还可以往回走...然后就不知道怎么做了... 看看了题解http://www.cnblogs.com/wuyiqi/archive/2 ...

随机推荐

  1. Ldap 从入门到放弃(一)

    OpenLDAP 2.4版本 快速入门 本文内容是自己通过官网文档.网络和相关书籍学习和理解并整理成文档,其中有错误或者疑问请在文章下方留言. 一.Introduction to OpenLDAP D ...

  2. ORM详解,ORM Object relation mapping (对象关系映射)

  3. Linux相关基础知识

    文件目录 /bin 放置系统执行档的目录,指令可被root与一般账户所使用. /boot 放置开机使用到的文档,包括linux核心档案,开机选单与所需设定档. /dev 任何装置与周边设备都是以档案的 ...

  4. [CSP-S模拟测试]:骆驼(模拟+构造)

    题目描述 我们都熟悉走马步,现在我们定义一种新的移动方式——骆驼步,它在一个国际棋盘上的移动规则是这样的. 以看出,骆驼步可以向八个方向走动,且不能走出棋盘范围. 现在给出一个$N\times N$的 ...

  5. React 进阶设计与控制权问题

    控制权--这个概念在编程中至关重要.比如,"轮子"封装层与业务消费层对于控制权的"争夺",就是一个很有意思的话题.这在 React 世界里也不例外.表面上看,我 ...

  6. MSSQL 如何导出db所有用户权限创建语句

    use dbgoDECLARE @Database varchar(255),@loginName varchar(255),@roleName varchar(255),@sql nvarchar( ...

  7. JavaVM & JNIEnv

    JNIEnv提供了大多数的JNI函数.你的本地方法都会接收JNIEnv作为第一个参数.JNIEnv用于本地线程存储.因此,你不能在线程间共享同一个JNIEnv.如果一个代码段没有其他方式获取它自身线程 ...

  8. React-Native 之 GD (八)GET 网络请求封装

    1.到这里,相信各位对 React-Native 有所熟悉了吧,从现在开始我们要慢慢往实际的方向走,这边就先从网络请求这部分开始,在正式开发中,网络请求一般都单独作为一部分,我们在需要使用的地方只需要 ...

  9. 在vue中如何引jquer

    最近学习vue,习惯性的通过<script>标签引入jquery,写完后报错才想起来,这种方式在vue是不适用的. 1:因为已经安装了vue脚手架,所以需要在webpack中全局引入jqu ...

  10. java开发客户端发送请求到服务器端出现这样:JSON parse error: Unexpected character ('}' (code 125)): was expecting

    org.springframework.http.converter.HttpMessageNotReadableException: JSON parse error: Unexpected cha ...