题目链接:https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&problem=4126

$LRJ$紫书例题$9-26$

题目大意:

给定一颗树 有些边已经标好方向 现在要给其余的边标上方向 使得最长的有向链最短

$HIT:$ 题目额外给了一个结论 假设已经确定方向的边所能得到的最长链为$k$ 最后的最长链一定$k$ 或$k + 1$

不知道是自己太久没有写树形$DP$还是这题的确比较麻烦 花了好久才折腾出来

首先 利用题目给的结论 我们实际上只需要解决以下这个几乎等价的问题

判断所给的树是否可以构造出最长链不超过$k$的方案

如果是的 那么最长链就是$k$否则为$k + 1$


对于这个可行解问题 我们可以这样考虑

对于每颗子树的最长链 它要么经过这颗子树的根节点 要么在这个根节点的某个儿子所对应的子树中

因此只需递归地去检验每颗子树是否合法即可

我们可以建立三个数组 $f[x][y]$ $up[x]$ $down[x]$

$f[x][y]$代表 到达该根节点$x$的向上的最长链最小值为$y$时向下的最长链最小值为多少

$up[x]$ $down[x]$分别代表该根节点向上/下的最长链的最小值

然后由于数据范围比较小 所以可以直接用这种$O(n^2)$的方法轻松解决

 #include <bits/stdc++.h>
using namespace std;
const int N = , E = N << ;
int firste[N], nexte[E], v[E], w[E];
int n, e, ans;
void build(int x, int y, int z)
{
nexte[++e] = firste[x];
firste[x] = e;
v[e] = y;
w[e] = z;
}
void init(int uu)
{
int vv;
char ch;
n = ;
e = ;
memset(firste, , sizeof firste);
ans = ;
do
{
n = max(n, uu);
while(scanf("%d%c", &vv, &ch), vv)
{
n = max(n, vv);
if(ch == 'd')
{
build(uu, vv, );
build(vv, uu, );
}
else if(ch == 'u')
{
build(uu, vv, );
build(vv, uu, );
}
else
{
build(uu, vv, );
build(vv, uu, );
}
}
}while(scanf("%d", &uu), uu);
}
void dfs0(int u, int fa, int dist)
{
ans = max(ans, dist);
for(int p = firste[u]; p; p = nexte[p])
if(w[p] && v[p] != fa)
dfs0(v[p], u, dist + );
}
int f[N][N], up[N], down[N];
bool dfs(int u, int fa)
{
for(int p = firste[u]; p; p = nexte[p])
if(v[p] != fa)
{
if(!dfs(v[p], u))
return ;
if(w[p])
{
for(int i = ; i <= min(down[v[p]], ans); ++i)
f[u][i] = ans + ;
}
else if(w[p ^ ])
{
for(int i = ; i <= ans; ++i)
f[u][i] = max(f[u][i], up[v[p]] + );
}
else
{
for(int i = ; i <= down[v[p]]; ++i)
f[u][i] = max(f[u][i], up[v[p]] + );
}
}
bool re = ;
for(int i = ; i <= ans; ++i)
if(f[u][i] + i <= ans)
{
re = ;
down[u] = min(down[u], i);
up[u] = min(up[u], f[u][i]);
}
return re;
}
int main()
{
int tmp;
while(scanf("%d", &tmp), tmp)
{
init(tmp);
for(int i = ; i <= n; ++i)
dfs0(i, i, );
memset(f, , sizeof f);
memset(down, 0x3f, sizeof down);
memset(up, 0x3f, sizeof up);
if(dfs(, ))
printf("%d\n", ans + );
else
printf("%d\n", ans + );
}
return ;
}

UVA 1380 A Scheduling Problem的更多相关文章

  1. 【暑假】[深入动态规划]UVa 1380 A Scheduling Problem

     UVa 1380 A Scheduling Problem 题目: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=41557 ...

  2. 【UVA 1380】 A Scheduling Problem (树形DP)

    A Scheduling Problem   Description There is a set of jobs, say x1, x2,..., xn <tex2html_verbatim_ ...

  3. UVa 101 The Blocks Problem Vector基本操作

    UVa 101 The Blocks Problem 一道纯模拟题 The Problem The problem is to parse a series of commands that inst ...

  4. UVA - 524 Prime Ring Problem(dfs回溯法)

    UVA - 524 Prime Ring Problem Time Limit:3000MS     Memory Limit:0KB     64bit IO Format:%lld & % ...

  5. uva 10837 - A Research Problem(欧拉功能+暴力)

    题目链接:uva 10837 - A Research Problem 题目大意:给定一个phin.要求一个最小的n.欧拉函数n等于phin 解题思路:欧拉函数性质有,p为素数的话有phip=p−1; ...

  6. UVA 810 - A Dicey Problem(BFS)

    UVA 810 - A Dicey Problem 题目链接 题意:一个骰子,给你顶面和前面.在一个起点,每次能移动到周围4格,为-1,或顶面和该位置数字一样,那么问题来了,骰子能不能走一圈回到原地, ...

  7. UVA 10026 Shoemaker's Problem 鞋匠的难题 贪心+排序

    题意:鞋匠一口气接到了不少生意,但是做鞋需要时间,鞋匠只能一双一双地做,根据协议每笔生意如果拖延了要罚钱. 给出每笔生意需要的天数和每天的罚钱数,求出最小罚钱的排列顺序. 只要按罚款/天数去从大到小排 ...

  8. UVA 1640 The Counting Problem UVA1640 求[a,b]或者[b,a]区间内0~9在里面各个数的数位上出现的总次数。

    /** 题目:UVA 1640 The Counting Problem UVA1640 链接:https://vjudge.net/problem/UVA-1640 题意:求[a,b]或者[b,a] ...

  9. Uva 101 -- the block problem

    Uva 101 the block problem 题目大意: 输入n,得到编号为0~n-1的木块,分别摆放在顺序排列编号为0~n-1的位置.现对这些木块进行操作,操作分为四种. 1.move a o ...

随机推荐

  1. JS观察者设计模式:实现iframe之间快捷通信

    观察者设计模式又称订阅发布模式,在JS中我们习惯叫做广播模式,当多个对象监听一个通道时,只要发布者向该通道发布命令,订阅者都可以收到该命令,然后执行响应的逻辑.今天我们要实现的就是通过观察者设计模式, ...

  2. MySQL总结(3)

    ORDER BY 与 GROUP BY ORDER BY GROUP BY 排序产生的输出 分组行.但输出可能不是分组的顺序 不一定需要 如果与聚集函数一起使用列(或表达式)则必须使用 任意列都可以使 ...

  3. 题解 AT2684 【K-City】

    此题这么水,竟然发题解的这么少. 本蒟蒻也来发一篇QwQ. 题目中所谓“四条街包围一个街区”其实就是两条街之间夹一个街区而已: n条街有几条两两相邻呢?答案是n-1条: m条街有几条两两相邻呢?答案是 ...

  4. selenium 教程

    selenium 本身是一套web自动化测试工具,但其经常被用于爬虫,解决一些复杂爬虫的问题. selenium 用于爬虫时,相当于模拟人操作浏览器. 浏览器驱动 使用 selenium 需要先安装 ...

  5. intellij idea 的快捷键方法

    1.Ctrl+N按名字搜索类 相当于eclipse的ctrl+shift+R,输入类名可以定位到这个类文件,就像idea在其它的搜索部分的表现一样,搜索类名也能对你所要搜索的内容多个部分进行匹配,而且 ...

  6. SPSS Statistics 26.0 下载安装和激活

    目录 1. 其他版本 2. IBM SPSS Statistics 26 新增功能 3. 安装步骤 4. 下载地址 1. 其他版本 参考:https://www.cnblogs.com/coco56/ ...

  7. SVN更新报错:Checksum mismatch for 解决办法

    问题: Checksum mismatch while updating '……'; expected: '3f9fd4dd7d1a0304d8020f73300a3e07', actual: 'cd ...

  8. Linux下安装chrome浏览器

    第一步:进入google-chrome官网下载chrome安装包 官网地址:https://www.google.cn/chrome/ 选择要下载的安装包 注意:这里有两个选项,请按照你安装的系统下载 ...

  9. 四、续绑定SignaIR的用户管理-(添加好友和消息盒子)

    一.聊天消息表(普通消息,申请消息,群聊消息) CREATE TABLE MSG_INFO ( MSG_Id INT PRIMARY KEY AUTO_INCREMENT, -- 消息标识 MSG_T ...

  10. 06java进阶——集合框架(list和泛型)

    1.ArrayList ArrayList集合是程序中最常见的一种集合,它属于引用数据类型(类).在ArrayList内部封装了一个长度可变的数组,当存入的元素超过数组长度时,ArrayList会在内 ...