ZOJ 3805--解题报告
题目相关:
3805相关链接: http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=5337
在二维的矩形上, 机器通过管道(pipe)连接(I型, L型),最终成为一个系统.
其规则大致提炼如下:
1). 编号大的输出可以成为编号小的输入(隐含拓扑序), 编号1只有输入/没有输出.
2). 每个节点最多有两个输入(弱化的图, 可以用二叉树来模拟)
3). 矩形世界没有高度限制, 但有宽度限制
目标就是:
最能满足要求的最小宽度是多少?
思路解析:
本题隐含拓扑序(有向图), 同时每个节点最多两个子节点, 因此我们采用二叉树去模拟. 这种最优化问题, 有可能是动态规划(树形DP), 通过观察和琢磨. 可得出如下结论:
如果节点i存在子节点j, k, 若节点j的宽度==节点k的宽度, 则取子节点宽度+1, 若子节点不相同, 则取子节点最大宽度的.
转化为公式如下:
{ node(j) + 1; if node(j) == node(k) --(1)
node(i) = {
{ max(node(j), node(k)); if node(j) != node(k) --(2)
其二叉树的是偏向树, 构建的时候, 往一个方向倾斜就是了.
AC代码:
#include <cstdio> #include <vector>
#include <stack> #include <functional> struct tree_node_t {
int left_index;
int right_index;
int value;
tree_node_t(int li = -1, int ri = -1, int v = -1)
: left_index(li), right_index(ri), value(v) {
}
}; class machine {
public:
machine() {
} // *) 输入数据并构建二叉树
void init_and_buildtree(int n, const std::vector<int> &datas) {
arr_tree.resize(n);
for ( int i = 1; i < datas.size(); i++ ) {
int idx = datas[i] - 1;
if ( arr_tree[idx].left_index == -1 ) {
arr_tree[idx].left_index = i;
} else {
arr_tree[idx].right_index = i;
}
}
} // *) 计算结果
// *) 这边采用模拟堆栈的方式, 来实现递归调用, 因为节点有10000个.
// *)最差情况会导致二叉树成链表, 导致递归的调用栈达到10000
int calculator() { std::stack<int> frames;
frames.push(0);
while ( !frames.empty() ) {
int idx = frames.top(); const int &li = arr_tree[idx].left_index;
const int &ri = arr_tree[idx].right_index; if ( li != -1 && arr_tree[li].value == -1 ) {
frames.push(li);
} else if ( ri != -1 && arr_tree[ri].value == -1 ) {
frames.push(ri);
} else {
if ( li == -1 ) {
arr_tree[idx].value = 1;
} else if ( li != -1 && ri == -1 ) {
arr_tree[idx].value = arr_tree[li].value;
} else if ( li != -1 && ri != -1 ) {
if ( arr_tree[li].value == arr_tree[ri].value ) {
arr_tree[idx].value = arr_tree[li].value + 1;
} else {
arr_tree[idx].value =
std::max(arr_tree[li].value, arr_tree[ri].value);
}
}
frames.pop();
} }
return arr_tree[0].value; } public:
std::vector<tree_node_t> arr_tree;
}; int main()
{ int n;
while ( scanf("%d", &n) != EOF ) {
std::vector<int> datas(n, 0);
for ( int i = 1; i < n; i++ ) {
scanf("%d", &datas[i]);
} machine instance;
instance.init_and_buildtree(n, datas); printf("%d\n", instance.calculator());
}
return 0; }
评注: 这边采用堆栈的方式来模拟递归调用, 是因为担心堆栈太深, 不过实际测试数据集, 没那么变态, 用递归的方式实现, 不仅优雅而且编码效率更高.
递归代码片段:
// *) 递归函数, 划分子问题
int evaluate(int idx) { int li = arr_tree[idx].left_index;
int ri = arr_tree[idx].right_index;
if ( li != -1 ) {
evaluate(li);
}
if ( ri != -1 ) {
evaluate(ri);
} if ( li == -1 ) {
return arr_tree[idx].value = 1;
} else if ( li != -1 && ri == -1 ) {
return arr_tree[idx].value = arr_tree[li].value;
} else {
if ( arr_tree[li].value == arr_tree[ri].value ) {
arr_tree[idx].value = arr_tree[li].value + 1;
} else {
arr_tree[idx].value = std::max(arr_tree[li].value, arr_tree[ri].value);
}
} return arr_tree[idx].value; } // *) 驱动函数
int calculator() {
return evaluate(0);
}
ZOJ 3805--解题报告的更多相关文章
- ZOJ 1093 Monkey and Banana (LIS)解题报告
ZOJ 1093 Monkey and Banana (LIS)解题报告 题目链接:http://acm.hust.edu.cn/vjudge/contest/view.action?cid= ...
- zoj 2313 Chinese Girls' Amusement 解题报告
题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1313 题目意思:有 N 个人(编号依次为1~N)围成一个圆圈,要求求 ...
- CH Round #56 - 国庆节欢乐赛解题报告
最近CH上的比赛很多,在此会全部写出解题报告,与大家交流一下解题方法与技巧. T1 魔幻森林 描述 Cortana来到了一片魔幻森林,这片森林可以被视作一个N*M的矩阵,矩阵中的每个位置上都长着一棵树 ...
- 二模13day1解题报告
二模13day1解题报告 T1.发射站(station) N个发射站,每个发射站有高度hi,发射信号强度vi,每个发射站的信号只会被左和右第一个比他高的收到.现在求收到信号最强的发射站. 我用了时间复 ...
- BZOJ 1051 最受欢迎的牛 解题报告
题目直接摆在这里! 1051: [HAOI2006]受欢迎的牛 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 4438 Solved: 2353[S ...
- 习题:codevs 2822 爱在心中 解题报告
这次的解题报告是有关tarjan算法的一道思维量比较大的题目(真的是原创文章,希望管理员不要再把文章移出首页). 这道题蒟蒻以前做过,但是今天由于要复习tarjan算法,于是就看到codevs分类强联 ...
- 习题:codevs 1035 火车停留解题报告
本蒟蒻又来写解题报告了.这次的题目是codevs 1035 火车停留. 题目大意就是给m个火车的到达时间.停留时间和车载货物的价值,车站有n个车道,而火车停留一次车站就会从车载货物价值中获得1%的利润 ...
- 习题: codevs 2492 上帝造题的七分钟2 解题报告
这道题是受到大犇MagHSK的启发我才得以想出来的,蒟蒻觉得自己的代码跟MagHSK大犇的代码完全比不上,所以这里蒟蒻就套用了MagHSK大犇的代码(大家可以关注下我的博客,友情链接就是大犇MagHS ...
- 习题:codevs 1519 过路费 解题报告
今天拿了这道题目练练手,感觉自己代码能力又增强了不少: 我的思路跟别人可能不一样. 首先我们很容易就能看出,我们需要的边就是最小生成树算法kruskal算法求出来的边,其余的边都可以删掉,于是就有了这 ...
- NOIP2016提高组解题报告
NOIP2016提高组解题报告 更正:NOIP day1 T2天天爱跑步 解题思路见代码. NOIP2016代码整合
随机推荐
- Prime Palindromes
题目大意:求出区间[a,b]之间的回文质数. a<=b<=10^8; 解题过程: 1.先打个素数表,新学了个 欧拉筛法,是对普通筛法的改进.普通筛法是每找到一个素数,就把它的所有倍数标记成 ...
- visual studio 2013连接Oracle 11g并获取数据:(二:实现)
1.VS中新建一个winform窗体 (1)一个按钮 (2)一个数据表格视图(在里面显示得到的数据表) 2.双击按钮进入代码 (1)添加 using System.Data.OracleClient; ...
- 如何在Linux上通过grub添加内核参数
转自Linux中国 我们可以在linux内核启动时为其提供各种各样的参数.这些参数可以自定义内核默认的行为,或者通知内核关于硬件的配置信息.内核参数应在内核启动时通过引导装载程序,如GRUB或LILO ...
- Section 1.4 Mother's Milk
又是一道怨念已久的题目0 0之前深搜写过广搜写过,怎么就是卡死,我还以为FP坏了重新装了一遍.今天偶尔翻起来,发现广搜忘记inc(head)了…简直哭瞎… 简单的广搜,分类比较多,不过不太要动脑子.至 ...
- hadoop源码阅读
1.Hadoop的包的功能分析 2.由于Hadoop的MapReduce和HDFS都有通信的需求,需要对通信的对象进行序列化.Hadoop并没有采用java的序列化,而是引入它自己的系统.org.ap ...
- URL详谈
URL(Uniform Resource Locator,统一资源定位符)是地址的别名.它包含关于文件存储位置和浏览器应如何处理它的信息.互联网上的每个文件都有唯一的 URL. URL 的第一个部分称 ...
- JAVA 打印九九乘法表
/** * * @author liangxiaoyu * @version 1.0 *2015-09-18 */public class JJ { public static void main( ...
- encodeURIComponent编码2次
最近在项目中发现前台在往anction中发送数据时进行两次encodeURIComponent,经过上网查找发现以下解释比较合理,mask过来,以供参考,非常感谢原作者!!! 两次encodeURIC ...
- Configuration for Python to run on Android
It's exiting to learn to make app on Android with python. But when I check <Head First Python> ...
- 刷固件Layer1到手机FLASH(硬刷)
开头: 注意:本文章并不是做GSM 嗅探必须的,平时我们刷机叫软刷是刷到内存里面的,断电就消失了,这个是硬刷,刷到flash里面的,断电不消失,开机就运行的. 本文章经过作者实测可行,这只是单个应用程 ...