zoukankan      html  css  js  c++  java
  • bzoj千题计划229:bzoj4424: Cf19E Fairy

    http://www.lydsy.com/JudgeOnline/problem.php?id=4424

    图是二分图的条件:没有奇环

    所以,如果图不存在奇环,删除任意一条边都可以

    如果存在奇环,

    对于树边来说:

    那么可能可以删除的边一定在所有奇环的交集内

    而且这条边不能在偶环内

    因为如果一条边既是奇环上的一条边,又是偶环上的一条边

    删除这条边后,这个奇环和偶环会合并成一个新的奇环

    所以最终的答案= 奇环的交集-偶环的并集

    对于非树边来说:

    如果只有一个奇环,那么可以删除构成环的这条非树边

     

    树边和非树边的判断:并查集

    奇环交与偶环并:差分,统计树上后缀和

    #include<cmath>
    #include<cstdio>
    #include<iostream>
    #include<algorithm>
    
    using namespace std;
    
    #define N 1000001
    
    int e[N][2],ty[N];
    
    int tot=1;
    int front[N],nxt[N<<1],to[N<<1],id[N<<1];
    
    int F[N];
    
    int maxd;
    int fa[N][21];
    
    int dep[N];
    
    int odd_cir,sum[N];
    
    int ans_tot,ans[N];
    
    void read(int &x)
    {
        x=0; char c=getchar();
        while(!isdigit(c)) c=getchar();
        while(isdigit(c)) { x=x*10+c-'0'; c=getchar(); }
    }
    
    int find(int i) { return F[i]==i ? i : F[i]=find(F[i]); }
    
    void add(int u,int v,int num)
    {
        to[++tot]=v; nxt[tot]=front[u]; front[u]=tot; id[tot]=num;
        to[++tot]=u; nxt[tot]=front[v]; front[v]=tot; id[tot]=num;
    }
    
    void dfs(int u)
    {
        for(int i=1;i<=maxd;++i) fa[u][i]=fa[fa[u][i-1]][i-1];
        int t;
        for(int i=front[u];i;i=nxt[i])
        {
            t=to[i];
            if(t==fa[u][0]) continue;
            dep[t]=dep[u]+1;
            fa[t][0]=u;
            dfs(t);
        }
    }
    
    int cal(int x)
    {
        for(int i=front[x];i;i=nxt[i])
            if(to[i]!=fa[x][0])
            {
                cal(to[i]);
                if(sum[to[i]]==odd_cir) ans[++ans_tot]=id[i];
                sum[x]+=sum[to[i]];
            }
    }
    
    int get_lca(int u,int v)
    {
        if(dep[u]<dep[v]) swap(u,v);
        int cnt=dep[u]-dep[v];
        for(int i=maxd;i>=0;--i)
            if(cnt&(1<<i)) u=fa[u][i];
        if(u==v) return u;
        for(int i=maxd;i>=0;--i)
            if(fa[u][i]!=fa[v][i]) u=fa[u][i],v=fa[v][i];
        return fa[u][0];
    }
    
    int main()
    {
        int n,m;
        int u,v,fu,fv;
        read(n); read(m);
        for(int i=1;i<=n;++i) F[i]=i;
        for(int i=1;i<=m;++i)
        {
            read(u); read(v);
            fu=find(u); fv=find(v);
            if(fu!=fv)
            {
                ty[i]=1;
                F[fu]=fv;
                add(u,v,i);
            }
            e[i][0]=u; e[i][1]=v;
        }
        maxd=log(n)/log(2);
        for(int i=1;i<=n;++i)
            if(find(i)==i) dep[i]=1,dfs(i);
        int self=0;
        int lca;
        for(int i=1;i<=m;++i)
        {        
            u=e[i][0]; v=e[i][1];
            if(u==v && self) { printf("0"); return 0; }
            if(u==v) { self=i; continue; }
            if(!ty[i])
            {
                lca=get_lca(u,v);
                if((dep[u]-dep[lca]+dep[v]-dep[lca])&1)
                {
                    sum[v]--;
                    sum[u]--;
                    sum[lca]+=2;
                }
                else
                {
                    ty[i]=2;
                    odd_cir++;
                    sum[v]++;
                    sum[u]++;
                    sum[lca]-=2;
                }
            }
        }
        if(self)
        {
            if(!odd_cir) printf("1
    %d",self);
            else printf("0");
            return 0;
        }
        if(!odd_cir)
        {
            printf("%d
    ",m);
            for(int i=1;i<m;++i) printf("%d ",i);
            if(m) printf("%d",m);
            return 0;
        }
        for(int i=1;i<=n;++i)
            if(find(i)==i) cal(i);
        if(odd_cir==1)
            for(int i=1;i<=m;++i)        
                if(ty[i]==2) 
                {
                    ans[++ans_tot]=i;
                    break;
                }
        sort(ans+1,ans+ans_tot+1);
        printf("%d
    ",ans_tot);
        for(int i=1;i<ans_tot;++i) printf("%d ",ans[i]);
        if(ans_tot) printf("%d",ans[ans_tot]);
    }

    4424: Cf19E Fairy

    Time Limit: 10 Sec  Memory Limit: 256 MB
    Submit: 740  Solved: 187
    [Submit][Status][Discuss]

    Description

    给定 n 个点,m 条边的无向图,可以从图中删除一条边,问删除哪些边可以使图变成
    一个二分图。

    Input

    第 1 行包含两个整数 n,m。分别表示点数和边数。
    第 2 到 m+1 行每行两个数 x,y 表示有一条(x,y)的边。

    Output

    输出第一行一个整数,表示能删除的边的个数。
    接下来一行按照从小到大的顺序输出边的序号。

    Sample Input

    4 4
    1 2
    1 3
    2 4
    3 4

    Sample Output

    4
    1 2 3 4

    HINT

    100%的数据,n,m<=1000000

  • 相关阅读:
    《Linux C编程一站式学习》第5章深入理解函数课后作业
    《Linux C编程一站式学习》——常量、变量和表达式notes
    《Linux C编程一站式学习》——第一个程序HelloWorld.c
    二分排序java实现
    leetcode初级算法(数组)——从数组中删除重复项
    利用层次遍历原理构建二叉树
    canvas学习笔记:绘制各种图形
    获取字符串长度【把双字节的替换成两个单字节的然后再获得长度
    oracle PLSQL程序造数据笔记
    oracle取一条记录中多个列的最大值和最小值
  • 原文地址:https://www.cnblogs.com/TheRoadToTheGold/p/8428557.html
Copyright © 2011-2022 走看看