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

题目1 : 彩色的树

时间限制:2000ms

单点时限:1000ms

内存限制:256MB

描述

给定一棵n个节点的树,节点编号为1, 2, …, n。树中有n - 1条边,任意两个节点间恰好有一条路径。这是一棵彩色的树,每个节点恰好可以染一种颜色。初始时,所有节点的颜色都为0。现在需要实现两种操作:

1. 改变节点x的颜色为y;

2. 询问整棵树被划分成了多少棵颜色相同的子树。即每棵子树内的节点颜色都相同,而相邻子树的颜色不同。

输入

第一行一个整数T,表示数据组数,以下是T组数据。

每组数据第一行是n,表示树的节点个数。接下来n - 1行每行两个数i和j,表示节点i和j间有一条边。接下来是一个数q,表示操作数。之后q行,每行表示以下两种操作之一:

1. 若为"1",则询问划分的子树个数。

2. 若为"2 x y",则将节点x的颜色改为y。

输出

每组数据的第一行为"Case #X:",X为测试数据编号,从1开始。

接下来的每一行,对于每一个询问,输出一个整数,为划分成的子树个数。

数据范围

1 ≤ T ≤ 20

0 ≤ y ≤ 100000

小数据

1 ≤ n, q ≤ 5000

大数据

1 ≤ n, q ≤ 100000

样例输入
2
3
1 2
2 3
3
1
2 2 1
1
5
1 2
2 3
2 4
2 5
4
1
2 2 1
2 3 2
1
样例输出
Case #1:
1
3
Case #2:
1
5
解答思路
本题主要解决任意时刻树中有多少棵颜色相同的子树,可以记录每一个节点的邻居节点,在每次更新一个节点的颜色的时候,都重新计算有多少棵同色子树。neighbor[i]表示节点i的邻居节点集合,color[i]表示节点i的颜色,new_color表示节点i新获得的颜色,color_tree_num表示树中有多少棵同色子树,初始为1。如果color[i] == new_color,color_tree_num不变;如果color[i] != new_color,节点获得了新颜色,需要更新color_tree_num  for(节点i的邻居节点 p: neighbor[i]){    if(p.color == color[i] /*&& p.color != new_color*/) color_tree_num++;    else if(/*p.color != color[i] &&*/ p.color == new_color) color_tree_num--;    /*else if(p.color != color[i] && p.color != new_color) color_tree_num不变*/  }
代码如下

#include<iostream>
#include<cstring>
#include<cstdio>
#include<unordered_set>
using namespace std;
int color[100010];
struct Node{
    //int i;
    int n;
    int *p;
};
Node neighbor[100010];
int main(){
    int T, n, q, i, j, query, c;
    unordered_set<long long> edge;
    memset(color, 0, sizeof(color));
    scanf("%d", &T);
    for(int k = 1; k <= T; k++){
        int tree = 1;
        printf("Case #%d:\n", k);
        scanf("%d", &n);
        for(int t = 1; t < n; t++){
            scanf("%d%d", &i, &j);
            color[i]++;
            color[j]++;
            long long ii = i, jj = j;
            if(i < j)
                edge.insert((ii << 32) | jj);
            else
                edge.insert((jj << 32) | ii);
        }
        for(i = 1; i <= n; i++){
            neighbor[i].n = 0;
            neighbor[i].p = new int[color[i]];
        }
        for(auto it = edge.begin(); it != edge.end(); ++it){
            j = int(*it);
            i = int((*it) >> 32);
            neighbor[i].p[neighbor[i].n++] = j;
            neighbor[j].p[neighbor[j].n++] = i;
        }

        memset(color, 0, sizeof(color));
        scanf("%d", &q);
        for(int t = 0; t < q; t++){
            scanf("%d", &query);
            if(query == 1){
                printf("%d\n", tree);
            }else{//query == 2
                scanf("%d%d", &i, &c);
                if(color[i] != c){
                    for(int* nei = neighbor[i].p, nn = neighbor[i].n, m = 0; m < nn; m++){
                        j = nei[m];
                        if(color[i] == color[j]){
                            tree++;
                        }else if(color[j] == c){
                            tree--;
                        }
                    }
                    color[i] = c;
                }
            }
        }
    }
    return 0;
}

时间: 2024-09-29 04:55:38

编程之美2015初赛第一场: 彩色的树的相关文章

编程之美2015初赛第一场 题解

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

hihocoder 编程之美2015初赛第一场 (树算法 + 暴力思想 + 搜索思想)

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

hiho 编程之美2015初赛第一场(质数相关-2分图匹配)

两个数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个整数,表示集合内的数. 输出 对于每组数据输出一行,形如"Case #X: Y"

编程之美2015初赛第一场BC

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

编程之美2015初赛第二场AB

题目1 : 扑克牌 时间限制:2000ms 单点时限:1000ms 内存限制:256MB 描述 一副不含王的扑克牌由52张牌组成,由红桃.黑桃.梅花.方块4组牌组成,每组13张不同的面值.现在给定52张牌中的若干张,请计算将它们排成一列,相邻的牌面值不同的方案数. 牌的表示方法为XY,其中X为面值,为2.3.4.5.6.7.8.9.T.J.Q.K.A中的一个.Y为花色,为S.H.D.C中的一个.如2S.2H.TD等. 输入 第一行为一个整数T,为数据组数. 之后每组数据占一行.这一行首先包含一个

hihocoder 微软编程之美2015 初赛 第二场(暴力+字典序+图论+思维算法)

题目1 : 扑克牌 时间限制:2000ms 单点时限:1000ms 内存限制:256MB 描述 一副不含王的扑克牌由52张牌组成,由红桃.黑桃.梅花.方块4组牌组成,每组13张不同的面值.现在给定52张牌中的若干张,请计算将它们排成一列,相邻的牌面值不同的方案数. 牌的表示方法为XY,其中X为面值,为2.3.4.5.6.7.8.9.T.J.Q.K.A中的一个.Y为花色,为S.H.D.C中的一个.如2S.2H.TD等. 输入 第一行为一个整数T,为数据组数. 之后每组数据占一行.这一行首先包含一个

编程之美2015初赛第二场 c题 八卦的小冰

题意:根据给出的各个用户之间的亲密度,计算所有异性对的亲密度总和. 思路:大数据怕太大,所以用邻接表保存这个无向图,在输入时顺便统计亲密度总和.对于更改性别:统计一下该用户的异性朋友的亲密度总和,再总计一下该用户同性朋友的亲密度总和,两者作差,再更新一下总亲密度.对于更改亲密度:要考虑改为0的情况,就是两用户没有关系了,图中删除这条边,根据亲密度的增加或减少,总亲密度也要相应增加或减少.要查询时直接输出.这样应该能应对全是3的情况.个人想法! 1 #include <iostream> 2 #

编程之美2015初赛_质数相关(二分图)

质数相关 题意:求二分图的最大独立集(独立集就是图中任意两个顶点都不相连的顶点集合) 思路:交替染色法,两种颜色把图染色.颜色最多的即为最大独立集. #include<iostream> #include<cstdio> #include<cstring> #include<cstdlib> #include<algorithm> #include<queue> #include<set> #include<map&

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

[来源] 题目2 : 树 [分析] 依据输入情况建立起树的模型.树的表示是一个表明父亲节点的数组.核心算法有两个: 计算某一节点的深度.用循环实现,一直向上找父亲节点,直到找到根节点.计算循环的次数即为深度. 计算某一节点的全部子节点.用递归实现. 本题在实现上节点的命名从0至N-1,与题目描写叙述不同. [代码] #include <iostream> #include <vector> using namespace std; vector<int> childre