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代码整合
随机推荐
- 终于!Linaro 加盟 Zephyr 项目
导读 为物联网构建实时操作系统的开源协作项目 Zephyr 项目宣布,Linaro 有限责任公司以白金会员的身份加盟该项目.Linaro是一家为 ARM 架构开发开源软件的协作工程组织,也是全球性机构 ...
- 数组越界保护与消息传递black机制
数组越界保护if(index.row <= [array count]) 发送消息[[NSNotificationCenter defaultCenter] postNotificati ...
- NSString asscii格式(2进制) 转 utf8格式——解决iOS自己处理http socket数据,遇到Transfer-Encoding: chunked时
因为需要实现自己的http客户端,就要自己模拟http 的socket通讯: 上行不难,自己处理好http即可. 但下行时,服务器端的动态语言返回数据有可能会是这种格式: http头 16进制表示的数 ...
- 2014北邮新生归来赛解题报告d-e
D: 399. Who Is Joyful 时间限制 3000 ms 内存限制 65536 KB 题目描述 There are several little buddies standing in a ...
- HDU 2795 单点更新,区间优先查找(想法)
Billboard Time Limit: 20000/8000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- 【C语言学习】-04 一维数组、字符数组
一维数组.数组排序.字符数
- Cisco IOS Debug Command Reference I through L
debug iapp through debug ip ftp debug iapp : to begin debugging of IAPP operations(in privileged EXE ...
- SharePoint 2013 开发——工作流架构
博客地址:http://blog.csdn.net/FoxDave SharePoint 2013的工作流较之前有了不同,第一次真正地作为独立的服务的概念推出了.这意味着SharePoint工作流不再 ...
- iOS 中 #import同@class之间的区别
很多刚开始学习iOS开发的同学可能在看别人的代码的时候会发现有部分#import操作写在m文件中,而h文件仅仅使用@class进行声明,不禁纳闷起来,为什么不直接把#import放到h文件中呢? 这是 ...
- android 单选、多选弹出菜单
菜单单选窗口: import android.app.Activity;import android.app.AlertDialog;import android.content.DialogInte ...