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

bzoj 4424: Cf19E Fairy && codeforces 19E. Fairy【树形dp】

时间:2018-07-22 23:42:56      阅读:176      评论:0      收藏:0      [点我收藏+]

标签:details   printf   dfs   iostream   getchar   turn   csdn   include   参考   

参考:https://blog.csdn.net/heheda_is_an_oier/article/details/51131641 这个找奇偶环的dp1真是巧妙,感觉像tarjan一样
首先分情况讨论,如果没有奇环,每条边都可以删;如果有一个奇环,奇环上隋边山;否则,删被所有奇环覆盖且没被任何一个偶环覆盖的边
那么重点就是怎样找到所有的奇环和偶环
用树形dp来搞,设f[i]记录经过第i条边的奇环数,g[i]记录经过第i条边的偶环数,因为是边的编号而存的是双向边,所以dp的时候用i>>1表示
然后随便dfs出一棵树,对于其他的返祖边如果是奇环的话f[i]++,偶环同理,并且加到父亲上
然后如果是返祖边的返祖边要减掉,因为已经统计过了

#include<iostream>
#include<cstdio>
using namespace std;
const int N=2000005;
int n,m,cnt=1,h[N],con,ans[N],f[N],g[N],p[N],q[N],top,tot;
bool v[N];
struct qwe
{
    int ne,to;
}e[N];
int read()
{
    int r=0,f=1;
    char p=getchar();
    while(p>‘9‘||p<‘0‘)
    {
        if(p==‘-‘)
            f=-1;
        p=getchar();
    }
    while(p>=‘0‘&&p<=‘9‘)
    {
        r=r*10+p-48;
        p=getchar();
    }
    return r*f;
}
void add(int u,int v)
{
    cnt++;
    e[cnt].ne=h[u];
    e[cnt].to=v;
    h[u]=cnt;
}
void dfs(int u,int fa)
{
    v[u]=1;
    p[u]=++top;
    for(int i=h[u];i;i=e[i].ne)
        if(q[i]!=-1)
        {
            if(!v[e[i].to])
            {
                q[i]=q[i^1]=-1;
                dfs(e[i].to,i>>1);
                f[fa]+=f[i>>1];
                g[fa]+=g[i>>1];
            }
            else
            {
                if(q[i]==1)
                    f[fa]--;
                if(q[i]==2)
                    g[fa]--;
                if(q[i]==0)
                {
                    if((p[u]-p[e[i].to])&1)
                        g[fa]++,q[i]=q[i^1]=2;
                    else
                        f[fa]++,q[i]=q[i^1]=1,con++;
                }
            }
        }
    top--;
}
int main()
{
    n=read(),m=read();
    for(int i=1;i<=m;i++)
    {
        int x=read(),y=read();
        add(x,y),add(y,x);
    }
    for(int i=1;i<=n;i++)
        if(!v[i])
            dfs(i,0);
    if(con==0)
    {
        tot=m;
        for(int i=1;i<=m;i++)
            ans[i]=i;
    }
    else
    {
        for(int i=1;i<=m;i++)
            if((f[i]==con&&g[i]==0)||(con==1&&q[i<<1]==1))
                ans[++tot]=i;
    }
    printf("%d\n",tot);
    for(int i=1;i<=tot;i++)
        printf("%d ",ans[i]);
    return 0;
}

bzoj 4424: Cf19E Fairy && codeforces 19E. Fairy【树形dp】

标签:details   printf   dfs   iostream   getchar   turn   csdn   include   参考   

原文地址:https://www.cnblogs.com/lokiii/p/9351741.html

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