码迷,mamicode.com
首页 > 其他好文 > 详细

Codeforces Round #395 (Div. 2) C

时间:2017-02-03 22:56:39      阅读:216      评论:0      收藏:0      [点我收藏+]

标签:大写   void   连接   stdio.h   i++   algorithm   ret   print   memset   

题意 : 给出一颗树 每个点都有一个颜色 选一个点作为根节点 使它的子树各自纯色

我想到了缩点后check直径 当<=3的时候可能有解 12必定有解 3的时候需要check直径中点的组成点里是否有一个点连接了另外所有的点 如果有就是ans 没有就是no

这个方法是对了 不过比较麻烦..需要很多check 

但是看div1的做法 有很棒的姿势用来解这个题

扫一下所有的边 如果两边的点颜色不一样 就增加一个连通分量数 这样可以很方便的算出有多少连通分量 同时记录这个点连着多少个其余的颜色(连通分量)

然后扫一下所有的点 看哪个点连着其余的连通分量 如果存在就是ans 没有就是no

果然还是应当多看一下别人的解法来学习 大写的服字 QAQ

我的迷茫解法 : 

#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<math.h>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define L long long
int n ;
vector<int >q[100050] ;
int c[100050] ;
int id[100050] ;
int gm[100050];
int cnt ;
void bfs(int s) {
    queue<int >que ;
    que.push(s) ;
    while(!que.empty()) {
        int u = que.front();
        que.pop();
        for(int i = 0; i <q[u].size(); i ++ ) {
            int v = q[u][i];
            if(id[v] == -1 && c[v] == c[u]) {
                id[v] = cnt ;
                gm[cnt] ++ ;
                que.push(v) ;
            }
        }
    }
}
vector<int >w[100050] ;
int vis[100050] ;
int res[100050];
int main(){
    scanf("%d",&n) ;for(int i = 1; i <=n;i++)q[i].clear();
    for(int i = 1; i < n ; i ++ ) {
        int u , v;
        scanf("%d%d",&u,&v);
        q[u].push_back(v);
        q[v].push_back(u);
    }
    for(int i = 1; i<=n;i++)scanf("%d",&c[i]) ;
    cnt = 0;
    memset(id,-1,sizeof(id)) ;
    memset(gm,0,sizeof(gm));
    for(int i = 1; i <= n ;i ++ ) {
        if(id[i]==-1){
                cnt ++ ;
            id[i] = cnt ;
            gm[cnt] = 1;
            bfs(i) ;
            res[cnt] = i ;
        }
    }
    if(cnt == 1) {
        printf("YES\n");
        printf("1\n");
        return 0 ;
    }
    if(cnt == 2) {
        printf("YES\n");
        for(int i = 1; i <= n ; i ++ ){
            for(int j = 0 ; j < q[i].size() ; j ++) {
                int v = q[i][j] ;
                if(c[i]!=c[v]) {
                    printf("%d\n",i);
                    return 0;
                }
            }
        }
    }
    for(int i = 1; i <= cnt ; i ++ )w[i].clear() ;
    for(int i = 1 ; i <= n ; i ++ ){
        for(int k = 0; k < q[i].size(); k ++ ){
            int v = q[i][k] ;
            if(id[v] != id[i]) {
                int uu = id[i] ;
                int vv = id[v] ;
                w[uu].push_back(vv);
            }
        }
    }
    int d = 0;
    int ans = 0;
    for(int i = 1; i <= cnt ;i ++ ){
        if(w[i].size() >= 2 ){
                d ++ ;
                if(d > 1){
                    ans = -1;
                    break ;
                }
                else {
                   ans = i ;
                }
        }
    }
    if(ans!= -1 && d == 1){
            int rr = -1;
            int cn = 0;
            for(int i = 1 ; i <= n ; i ++ ) {
                if(id [i] == ans ){
                    int bb = 0;
                    for (int j = 0 ; j < q[i].size(); j ++ ){
                        int  v = q[i][j] ;
                        if(id[v] != id[i])bb ++ ;
                    }
                    if(bb == w[ans].size()) {
                        rr = i ;
                        cn ++ ;
                    }
                }
            }
            if(cn == 1) {
                printf("YES\n");
                printf("%d\n",rr) ;
            }
            else printf("NO\n") ;
    }
    else printf("NO\n") ;
}

学习的姿势 : 

#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<math.h>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define L long long
int x[100050] ;
int y[100050] ;
int c[100050] ;
int n ;
int a[100050] ;
int main(){
    scanf("%d",&n) ;
    for(int i = 1; i < n ; i ++ ){
        scanf("%d%d",&x[i] , &y[i]) ;
    }
    for(int i = 1; i <= n ; i ++ ){
        scanf("%d",&c[i]) ;
    }
    memset(a , 0 , sizeof(a)) ;
    int b = 0 ;
    for(int i = 1 ; i < n ; i ++ ){
        if(c[x[i]] != c[y[i]]) {
            b ++ ;
            a[x[i]] ++ ;
            a[y[i]] ++ ;
        }
    }
    int ans = -1 ;
    for(int i = 1 ; i <= n ; i ++ ){
        if(a[i] == b) {
            ans = i;
            break ;
        }
    }
    if(ans == -1 )printf("NO\n") ;
    else {
        printf("YES\n");
        printf("%d\n",ans) ;
    }
}

  

Codeforces Round #395 (Div. 2) C

标签:大写   void   连接   stdio.h   i++   algorithm   ret   print   memset   

原文地址:http://www.cnblogs.com/rayrayrainrain/p/6363600.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!