微软2014编程之美初赛第一场——题目2 : 树

【来源】

题目2 : 树

【分析】

依据输入情况建立起树的模型。树的表示是一个表明父亲节点的数组。核心算法有两个:

  1. 计算某一节点的深度。用循环实现,一直向上找父亲节点,直到找到根节点。计算循环的次数即为深度。
  2. 计算某一节点的全部子节点。用递归实现。

本题在实现上节点的命名从0至N-1,与题目描写叙述不同。

【代码】

#include <iostream>
#include <vector>
using namespace std;

vector<int> children(int node, int* father, int n)
{
    vector<int> res;
    for (int i = 0; i < n; ++i){
        if (father[i] == node){
            res.push_back(i);
            vector<int> cs = children(i, father, n);
            if (cs.size() != 0){
                for (int j = 0; j < cs.size(); ++j){
                    res.push_back(cs[j]);
                }
            }
        }
    }

    return res;
}
int main()
{
    int T;
    cin >> T;
    for (int c = 0; c < T; ++c){
        int N;
        cin >> N;
        int* w = new int[N];
        for (int i = 0; i < N; ++i){
            w[i] = 0;
        }
        int* father = new int[N];
        for (int i = 1; i < N; ++i){
            cin >> father[i];
            father[i]--;
        }
        // build the tree
        int* dep = new int[N];
        dep[0] = 1;
        for (int i = 1; i < N; ++i){
            int d = 1;
            int node = i;
            while (node != 0){
                node = father[node];
                ++d;
            }
            dep[i] = d;
        }

        int Q;
        cin >> Q;
        for (int i = 0; i < Q; ++i){
            int u, l, r, delta;
            cin >> u >> l >> r >> delta;
            //do the operation
            vector<int> childs = children(u-1, father, N);

            for (int j = 0; j < childs.size(); ++j){
                if (dep[childs[j]] >= l && dep[childs[j]] <= r){
                    w[childs[j]] += delta;
                }
            }

        }

        //calc hash
        int MOD = 1000000007;
        int MAGIC = 12347;
        int Hash = 0;
        for (int i = 0; i < N; ++i){
            Hash = (Hash*MAGIC + w[i]) % MOD;
        }
        //output
        cout << "Case " << c+1 << ": " << Hash << endl;
    }

    system("pause");
    return 0;
}

【点评】

本题考查树的相关知识以及递归程序的设计。重点在于计算某个节点的深度和计算某个节点的全部子节点。

【备注】

本题做完后刚准备提交比赛就结束了= =,没有机会提交验证答案了T T。代码写的非常混乱,有问题欢迎留言探讨。

微软2014编程之美初赛第一场——题目2 : 树

时间: 2024-10-21 21:24:27

微软2014编程之美初赛第一场——题目2 : 树的相关文章

微软2014编程之美初赛第一场——题目3 : 活动中心

[来源] 题目3 : 活动中心 [分析] 本题採用的是三分法. 输入的一组点中找出左右边界.作为起始边界. while(右边界-左边界<精度){ 将左右边界构成的线段均匀分成3段,推断切割点的距离关系,抹去距离大的一段.更新左右边界. } 输出左(右)边界 [代码] #include <iostream> #include <vector> #include <cmath> #include <iomanip> using namespace std;

微软2014编程之美初赛第二场——题目2 : 字符串压缩

[来源] 题目2 : 字符串压缩 [分析] 把游程编码恢复为原始字符串,然后得出每一行的字符串的内容,放在一个vector中.用map统计vector中每一行的反复的次数. 比較两个游程编码得到的map是否同样就可以. 该算法占用空间太多,小数据AC,大数据MLE了. [代码] #include <iostream> #include <string> #include <vector> #include <map> #include <sstream

编程之美初赛第一场 树

题目2 : 树 时间限制:4000ms 单点时限:2000ms 内存限制:256MB 描写叙述 有一个N个节点的树.当中点1是根.初始点权值都是0. 一个节点的深度定义为其父节点的深度+1,.特别的.根节点的深度定义为1. 如今须要支持一系列下面操作:给节点u的子树中,深度在l和r之间的节点的权值(这里的深度依旧从整个树的根节点開始计算).都加上一个数delta. 问完毕全部操作后,各节点的权值是多少. 为了降低巨大输出带来的开销.如果完毕全部操作后.各节点的权值是answer[1..N],请你

2014微软编程之美初赛第一场第三题 活动中心

活动中心 时间限制:12000ms 单点时限:6000ms 内存限制:256MB 描写叙述 A市是一个高度规划的城市,可是科技高端发达的地方,居民们也不能忘记运动和锻炼,因此城市规划局在设计A市的时候也要考虑为居民们建造一个活动中心,方便居住在A市的居民们能随时开展运动,锻炼强健的身心. 城市规划局希望活动中心的位置满足下面条件: 1. 到全部居住地的总距离最小. 2. 为了方便活动中心的资源补给和其它器材的维护,活动中心必须建设在A市的主干道上. 为了简化问题.我们将A市摆在二维平面上,城市的

编程之美初赛第一场--焦距

题目1 : 焦距 时间限制:2000ms 单点时限:1000ms 内存限制:256MB 描写叙述 一般来说.我们採用针孔相机模型,也就是觉得它用到的是小孔成像原理. 在相机坐标系下,一般来说,我们用到的单位长度.不是"米"这种国际单位,而是相邻像素的长度.而焦距在相机坐标系中的大小,是在图像处理领域的一个很重要的物理量. 如果我们已经依据相机參数,得到镜头的物理焦距大小(focal length),和相机胶片的宽度(CCD width),以及照片的横向分辨率(image width),

编程之美初赛第一场

题目3 : 活动中心 时间限制:12000ms 单点时限:6000ms 内存限制:256MB 描写叙述 A市是一个高度规划的城市.可是科技高端发达的地方,居民们也不能忘记运动和锻炼,因此城市规划局在设计A市的时候也要考虑为居民们建造一个活动中心.方便居住在A市的居民们能随时开展运动.锻炼强健的身心. 城市规划局希望活动中心的位置满足下面条件: 1. 到全部居住地的总距离最小. 2. 为了方便活动中心的资源补给和其它器材的维护,活动中心必须建设在A市的主干道上. 为了简化问题.我们将A市摆在二维平

2015编程之美初赛第一场 A 彩色的树

时间限制:2000ms 单点时限:1000ms 内存限制:256MB 描述 给定一棵n个节点的树,节点编号为1, 2, -, n.树中有n - 1条边,任意两个节点间恰好有一条路径.这是一棵彩色的树,每个节点恰好可以染一种颜色.初始时,所有节点的颜色都为0.现在需要实现两种操作: 1. 改变节点x的颜色为y: 2. 询问整棵树被划分成了多少棵颜色相同的子树.即每棵子树内的节点颜色都相同,而相邻子树的颜色不同. 输入 第一行一个整数T,表示数据组数,以下是T组数据. 每组数据第一行是n,表示树的节

2015编程之美初赛第一场 B 建造金字塔

 时间限制:4000ms 单点时限:2000ms 内存限制:256MB 描述 在二次元中,金字塔是一个底边在x轴上的等腰直角三角形. 你是二次元世界的一个建筑承包商.现在有N个建造订单,每个订单有一个收益w,即建造此金字塔可获得w的收益.对每个订单可以选择建造或不建造. 建造一个金字塔的成本是金字塔的面积,如果两个或多个金字塔有重叠面积,则建造这些金字塔时重叠部份仅需建造一次. 建造一组金字塔的总利润是收益总和扣除成本.现给出这些订单,请求出最大利润. 输入 输入数据第一行为一个整数T,表示

2015编程之美初赛第一场 C 质数相关

 时间限制:2000ms 单点时限:1000ms 内存限制:256MB 描述 两个数a和 b (a<b)被称为质数相关,是指a × p = b,这里p是一个质数.一个集合S被称为质数相关,是指S中存在两个质数相关的数,否则称S为质数无关.如{2, 8, 17}质数无关,但{2, 8, 16}, {3, 6}质数相关.现在给定一个集合S,问S的所有质数无关子集中,最大的子集的大小. 输入 第一行为一个数T,为数据组数.之后每组数据包含两行. 第一行为N,为集合S的大小.第二行为N个整数,表示集