题目相关:
  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--解题报告的更多相关文章

  1. ZOJ 1093 Monkey and Banana (LIS)解题报告

    ZOJ  1093   Monkey and Banana  (LIS)解题报告 题目链接:http://acm.hust.edu.cn/vjudge/contest/view.action?cid= ...

  2. zoj 2313 Chinese Girls' Amusement 解题报告

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1313 题目意思:有 N 个人(编号依次为1~N)围成一个圆圈,要求求 ...

  3. CH Round #56 - 国庆节欢乐赛解题报告

    最近CH上的比赛很多,在此会全部写出解题报告,与大家交流一下解题方法与技巧. T1 魔幻森林 描述 Cortana来到了一片魔幻森林,这片森林可以被视作一个N*M的矩阵,矩阵中的每个位置上都长着一棵树 ...

  4. 二模13day1解题报告

    二模13day1解题报告 T1.发射站(station) N个发射站,每个发射站有高度hi,发射信号强度vi,每个发射站的信号只会被左和右第一个比他高的收到.现在求收到信号最强的发射站. 我用了时间复 ...

  5. BZOJ 1051 最受欢迎的牛 解题报告

    题目直接摆在这里! 1051: [HAOI2006]受欢迎的牛 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4438  Solved: 2353[S ...

  6. 习题:codevs 2822 爱在心中 解题报告

    这次的解题报告是有关tarjan算法的一道思维量比较大的题目(真的是原创文章,希望管理员不要再把文章移出首页). 这道题蒟蒻以前做过,但是今天由于要复习tarjan算法,于是就看到codevs分类强联 ...

  7. 习题:codevs 1035 火车停留解题报告

    本蒟蒻又来写解题报告了.这次的题目是codevs 1035 火车停留. 题目大意就是给m个火车的到达时间.停留时间和车载货物的价值,车站有n个车道,而火车停留一次车站就会从车载货物价值中获得1%的利润 ...

  8. 习题: codevs 2492 上帝造题的七分钟2 解题报告

    这道题是受到大犇MagHSK的启发我才得以想出来的,蒟蒻觉得自己的代码跟MagHSK大犇的代码完全比不上,所以这里蒟蒻就套用了MagHSK大犇的代码(大家可以关注下我的博客,友情链接就是大犇MagHS ...

  9. 习题:codevs 1519 过路费 解题报告

    今天拿了这道题目练练手,感觉自己代码能力又增强了不少: 我的思路跟别人可能不一样. 首先我们很容易就能看出,我们需要的边就是最小生成树算法kruskal算法求出来的边,其余的边都可以删掉,于是就有了这 ...

  10. NOIP2016提高组解题报告

    NOIP2016提高组解题报告 更正:NOIP day1 T2天天爱跑步 解题思路见代码. NOIP2016代码整合

随机推荐

  1. Asynchttpclient开源框架下载图片和文本,于Volley和Glide开源框架的区别。

    AsyncHttpClient是一款比较流行的Android异步网路加载库,在github上的网址是:https://github.com/loopj/android-async-httpAsyncH ...

  2. MySQL为数据表的指定字段插入数据

    username not null 没有默认值/有默认值   insert不插入username字段 均不报错 2014年07月23日21:05    百科369 MySQL为数据表的指定字段插入数据 ...

  3. jsp中JavaBean的用法

    UserRegisterBean.java:这是JavaBean package JavaBean; public class UserRegisterBean { private String us ...

  4. jQuery 其他操作

    1. 样式操作 1.1 获取 class 和设置 class : class 是元素的一个属性, 所以获取 class 和设置 class 都可以使用 attr() 方法来完成. 1.2 追加样式: ...

  5. struts中的常量,action配置中的默认值

    1.struts中Action的开发方式 继承ActionSupport类,这种方法实现的Action可以进行数据校验: 实现Action接口: 不继承任何类,不实现任何接口: 是否继承类或实现接口, ...

  6. Dom操作html详细

    <p name='pn'>xxx</p> <p name='pn'>xxx</p> <p name='pn'>xxx</p> & ...

  7. (转)深入理解javascript的function

    原文:http://www.cnblogs.com/sharpxiajun/archive/2011/09/16/2179323.html javascript笔记:深入理解javascript的fu ...

  8. ant新建scp和sshexec任务

    1.build.xml中新建targer如下: <target name="remotecopytest" description="拷贝文件到远程服务器" ...

  9. 【LeetCode OJ】Populating Next Right Pointers in Each Node II

    Problem Link: http://oj.leetcode.com/problems/populating-next-right-pointers-in-each-node-ii/ OK... ...

  10. 简单的多线程(活用OD解决运行时错误)

    代码源自<VC++深入详解>第15章 “多线程”,位于第563页 - 566 页. 程序的目的是展示多线程运行的效果: #include <windows.h> #includ ...