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);
}

  

时间: 2024-10-22 01:12:42

ZOJ 3805--解题报告的相关文章

ZOJ 3706 Break Standard Weight 解题报告

题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=5009 题目意思:给出两个mass:x 和 y,问如何将其中一个 mass 一分为二(当然分完之后它们的和要等于原来的mass,或x 或 y),使得利用这三个mass 可称的数量最大.输出这个最大数量. 网上参考别人用STL中的set来写,太厉害了!!!考虑到set对于重复的元素只存储一个,那么当三个mass组合的过程中有重复的,它都会自动舍弃有重复的,不需要用if来

zoj 2313 Chinese Girls&#39; Amusement 解题报告

题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1313 题目意思:有 N 个人(编号依次为1~N)围成一个圆圈,要求求出最大的 K (1 ≤ K ≤ N/2),表示从编号为1的人开始,将球传递给他后一个人数起的第K个人,第K个人又传递给往后数的第K个人......要求这样传递下去,且每个人都有机会接到球.也就是不存在当未使得全部人都接到一次球的情况下,某个人接收到两次以上的球. 详细的解题报告在这里: http:/

ZOJ Monthly, June 2014 解题报告

A.Another Recurrence Sequence B.Gears 题目大意:有n个齿轮,一开始各自为一组,之后进行m次操作,包括以下4种类型: 1.合并两组齿轮,合并的两个应该反向旋转 2.把某个齿轮从所在组删除,自为一组,但不影响同组其它齿轮的状态与关系 3.询问两个齿轮是同向.反向或无关系(即不在同一组) 4.询问某个齿轮所在组的齿轮总数 分析:典型的并查集操作,但是注意两点: 1.由于操作3要询问两个齿轮的相对状态,因此对并查集中每个元素应当保存它的状态信息.状态是相对的,只需要

解题报告 之 SOJ2414 Leapin&#39; Lizards

解题报告 之 SOJ2414 Leapin' Lizards Description Your platoon of wandering lizards has entered a strange room in the labyrinth you are exploring. As you are looking around for hidden treasures, one of the rookies steps on an innocent-looking stone and the

解题报告 之 POJ2135 Farm Tour

解题报告 之 POJ2135 Farm Tour Description When FJ's friends visit him on the farm, he likes to show them around. His farm comprises N (1 <= N <= 1000) fields numbered 1..N, the first of which contains his house and the Nth of which contains the big barn.

解题报告 之 POJ3057 Evacuation

解题报告 之 POJ3057 Evacuation Description Fires can be disastrous, especially when a fire breaks out in a room that is completely filled with people. Rooms usually have a couple of exits and emergency exits, but with everyone rushing out at the same time

hdu 1541 Stars 解题报告

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1541 题目意思:有 N 颗星星,每颗星星都有各自的等级.给出每颗星星的坐标(x, y),它的等级由所有比它低层(或者同层)的或者在它左手边的星星数决定.计算出每个等级(0 ~ n-1)的星星各有多少颗. 我只能说,题目换了一下就不会变通了,泪~~~~ 星星的分布是不是很像树状数组呢~~~没错,就是树状数组题来滴! 按照题目输入,当前星星与后面的星星没有关系.所以只要把 x 之前的横坐标加起来就可以了

【百度之星2014~初赛(第二轮)解题报告】Chess

声明 笔者最近意外的发现 笔者的个人网站http://tiankonguse.com/ 的很多文章被其它网站转载,但是转载时未声明文章来源或参考自 http://tiankonguse.com/ 网站,因此,笔者添加此条声明. 郑重声明:这篇记录<[百度之星2014~初赛(第二轮)解题报告]Chess>转载自 http://tiankonguse.com/ 的这条记录:http://tiankonguse.com/record/record.php?id=667 前言 最近要毕业了,有半年没做

2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告

2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告 勘误1:第6题第4个 if最后一个条件粗心写错了,答案应为1580. 条件应为abs(a[3]-a[7])!=1,宝宝心理苦啊.!感谢zzh童鞋的提醒. 勘误2:第7题在推断连通的时候条件写错了,后两个if条件中是应该是<=12 落了一个等于号.正确答案应为116. 1.煤球数目 有一堆煤球.堆成三角棱锥形.详细: 第一层放1个, 第二层3个(排列成三角形), 第三层6个(排列成三角形), 第四层10个(排列成三角形). -. 假设一共

[noip2011]铺地毯(carpet)解题报告

最近在写noip2011的题,备战noip,先给自己加个油! 下面是noip2011的试题和自己的解题报告,希望对大家有帮助,题目1如下 1.铺地毯(carpet.cpp/c/pas) [问题描述]为了准备一个独特的颁奖典礼,组织者在会场的一片矩形区域(可看做是平面直角坐标系的第一象限)铺上一些矩形地毯.一共有n 张地毯,编号从1 到n.现在将这些地毯按照编号从小到大的顺序平行于坐标轴先后铺设,后铺的地毯覆盖在前面已经铺好的地毯之上.地毯铺设完成后,组织者想知道覆盖地面某个点的最上面的那张地毯的