zoukankan      html  css  js  c++  java
  • C. Bakry and Partitioning 题解(思维+两次dfs)

    题目链接

    题目思路

    这个题目的思路还是没有那么的难

    首先如果要分成偶数个块,那么这个树的总异或和一定要为0,并且如果总异或和为0,那么必定可以分为两个联

    通块,如果分为奇数个的话,那么就看是否可以分为3个即可,就是先删除一个深度最大的联通块的异或和为总异

    或和,然后再dfs判断是否还有另外一个,没想到居然就是两次dfs的事情就行了,主要是没想到先删除

    代码

    #include<bits/stdc++.h>
    #define pii pair<int,int>
    #define fi first
    #define se second
    #define debug cout<<"I AM HERE"<<endl;
    using namespace std;
    typedef long long ll;
    const int maxn=1e5+5,inf=(1ll<<31)-1,mod=1e9+7;
    const double eps=1e-6;
    int n,k;
    vector<int> g[maxn];
    int dp[maxn];
    int a[maxn];
    int dep[maxn];
    bool vis[maxn];
    void dfs(int u,int fa){
        dp[u]=a[u];
        dep[u]=dep[fa]+1;
        for(auto x:g[u]){
            if(x==fa) continue;
            if(vis[x]) continue;
            dfs(x,u);
            dp[u]^=dp[x];
        }
    }
    signed main(){
        int _;scanf("%d",&_);
        while(_--){
            scanf("%d%d",&n,&k);
            int sum=0;
            for(int i=1;i<=n;i++){
                scanf("%d",&a[i]);
                sum^=a[i];
                dep[i]=0;
                g[i].clear();
            }
            for(int i=1,u,v;i<=n-1;i++){
                scanf("%d%d",&u,&v);
                g[u].push_back(v);
                g[v].push_back(u);
            }
            dfs(1,1);
            bool flag=0;
            int ma=-1,id=-1;
            for(int i=2;i<=n;i++){
                if(dp[i]==sum&&dep[i]>ma){
                    id=i;
                    ma=dep[i];
                }
            }
            if(dp[1]==0){
                flag=1;
            }else{
                if(id!=-1&&k>=3){
                    for(int i=1;i<=n;i++){
                        dp[i]=0;
                    }
                    vis[id]=1;
                    dfs(1,1);
                    for(int i=2;i<=n;i++){
                        if(dp[i]==sum){
                            flag=1;
                            break;
                        }
                        if(i==n){
                            flag=0;
                        }
                    }
                    vis[id]=0;
                }
            }
            printf(flag?"YES
    ":"NO
    ");
        }
        return 0;
    }
    
    
    
    不摆烂了,写题
  • 相关阅读:
    扫描线算法
    评论备份(3)
    评论备份(2)
    二分法的注意事项
    sam模板
    Machine Learning(Andrew Ng)学习笔记
    洛谷P2221 [HAOI2012]高速公路
    洛谷P3233 [HNOI2014]世界树
    P2515 [HAOI2010]软件安装
    BZOJ4293: [PA2015]Siano
  • 原文地址:https://www.cnblogs.com/hunxuewangzi/p/15366707.html
Copyright © 2011-2022 走看看