E. Fairy
time limit per test

1.5 seconds

memory limit per test

256 megabytes

input

standard input

output

standard output

Once upon a time there lived a good fairy A. One day a fine young man B came to her and asked to predict his future. The fairy looked into her magic ball and said that soon the fine young man will meet the most beautiful princess ever and will marry her. Then she drew on a sheet of paper n points and joined some of them with segments, each of the segments starts in some point and ends in some other point. Having drawn that picture, she asked the young man to erase one of the segments from the sheet. Then she tries to colour each point red or blue so, that there is no segment having points of the same colour as its ends. If she manages to do so, the prediction will come true. B wants to meet the most beautiful princess, that's why he asks you to help him. Find all the segments that will help him to meet the princess.

Input

The first input line contains two integer numbers: n — amount of the drawn points and m — amount of the drawn segments (1 ≤ n ≤ 104, 0 ≤ m ≤ 104). The following m lines contain the descriptions of the segments. Each description contains two different space-separated integer numbers vu (1 ≤ v ≤ n, 1 ≤ u ≤ n) — indexes of the points, joined by this segment. No segment is met in the description twice.

Output

In the first line output number k — amount of the segments in the answer. In the second line output k space-separated numbers — indexes of these segments in ascending order. Each index should be output only once. Segments are numbered from 1 in the input order.

Examples
input

Copy
4 4
1 2
1 3
2 4
3 4
output
4
1 2 3 4
input

Copy
4 5
1 2
2 3
3 4
4 1
1 3
output
1
5
题目大意:给定一个无向图,问删哪些边之后这个图变成二分图。求的是一个边的集合,实际上只删一条边.
分析:挺好的一道题.
   一个图是二分图的充要条件是不存在奇环.将一条奇环上的边删去就能破坏掉这个奇环,如果要破坏掉所有的奇环,那么删的边就必须是所有奇环的交集.
   仅仅只是删掉交集这么简单吗?如果一条边同时在偶环和奇环上,删掉这条边后偶环和奇环就会重新组合成一个奇环.那么删的这条边就必须满足两个条件:1.在所有奇环的交集中. 2.不在任何偶环上.
   那么找环就好了.天真的我以为直接dfs+栈维护一下就好了.这道题的环是会重叠的,这种做法行不通......换一种做法,每个点记录第一条连向这个点的边的编号(其实记录的就是树边),那么可以把边转换为点,在点上对树边进行操作,非树边需要特判一下.
   每次找到一条非树边,这条边连接的两个点是u,v,如果构成了一个奇环,就在维护奇环线段树中把u,v这条链+1,否则在维护偶环线段树中把u,v这条链+1.怎么提取这条链?树链剖分!
   最后是一些细节:如果没有奇环,所有的边都满足条件;如果奇环只有1个,那么那个奇环的非树边要考虑进来;维护的是一个图而不是树,在dfs时只考虑树边!
#include <stack>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm> using namespace std; const int maxn = ; int n,m,head[maxn],to[maxn],nextt[maxn],id[maxn],tot = ,vis[maxn],cnta,ans,cntb,anss[maxn];
int bianhao[maxn],h[maxn],son[maxn],top[maxn],pos[maxn],sizee[maxn],cnt,fa[maxn],flag;
int L[maxn << ],R[maxn << ],sum1[maxn << ],sum2[maxn << ],tag1[maxn << ],tag2[maxn << ]; struct node
{
int x,y;
}e[maxn]; void add(int x,int y)
{
to[tot] = y;
nextt[tot] = head[x];
head[x] = tot++;
} void dfs1(int u,int faa)
{
h[u] = h[faa] + ;
sizee[u] = ;
fa[u] = faa;
for (int i = head[u];i;i = nextt[i])
{
int v = to[i];
if (v == faa || h[v])
continue;
bianhao[v] = (i / ) + (i % );
vis[(i / ) + (i % )] = ;
dfs1(v,u);
sizee[u] += sizee[v];
if (sizee[v] > sizee[son[u]])
son[u] = v;
}
} void dfs2(int u,int topp)
{
pos[u] = ++cnt;
id[cnt] = u;
top[u] = topp;
if (son[u])
dfs2(son[u],topp);
for (int i = head[u];i;i = nextt[i])
{
int v = to[i];
if (v == fa[u] || v == son[u] || fa[v] != u)
continue;
dfs2(v,v);
}
} void build(int o,int l,int r)
{
L[o] = l;
R[o] = r;
if (l == r)
return;
int mid = (l + r) >> ;
build(o * ,l,mid);
build(o * + ,mid + ,r);
} void pushup(int o)
{
sum1[o] = sum1[o * ] + sum1[o * + ];
sum2[o] = sum2[o * ] + sum2[o * + ];
} void pushdown(int o)
{
if (tag1[o])
{
tag1[o * ] += tag1[o];
tag1[o * + ] += tag1[o];
sum1[o * ] += tag1[o] * (R[o * ] - L[o * ] + );
sum1[o * + ] += tag1[o] * (R[o * + ] - L[o * + ] + );
tag1[o] = ;
}
if (tag2[o])
{
tag2[o * ] += tag2[o];
tag2[o * + ] += tag2[o];
sum2[o * ] += tag2[o] * (R[o * ] - L[o * ] + );
sum2[o * + ] += tag2[o] * (R[o * + ] - L[o * + ] + );
tag2[o] = ;
}
} void update1(int o,int l,int r,int x,int y)
{
if(x <= l && r <= y)
{
sum1[o] += (r - l + );
tag1[o]++;
return;
}
pushdown(o);
int mid = (l + r) >> ;
if (x <= mid)
update1(o * ,l,mid,x,y);
if (y > mid)
update1(o * + ,mid + ,r,x,y);
} void update2(int o,int l,int r,int x,int y)
{
if(x <= l && r <= y)
{
sum2[o] += (r - l + );
tag2[o]++;
return;
}
pushdown(o);
int mid = (l + r) >> ;
if (x <= mid)
update2(o * ,l,mid,x,y);
if (y > mid)
update2(o * + ,mid + ,r,x,y);
} void change(int x,int y,int tagg)
{
if (h[x] < h[y])
swap(x,y);
while (top[x] != top[y])
{
if (h[top[x]] < h[top[y]])
swap(x,y);
int t = top[x];
if (tagg == )
update1(,,n,pos[t],pos[x]);
else
update2(,,n,pos[t],pos[x]);
x = fa[t];
}
if (x == y)
return;
if (h[x] < h[y])
swap(x,y);
if (tagg == )
update1(,,n,pos[y] + ,pos[x]); //为什么要+1?因为实际维护的是边.
else
update2(,,n,pos[y] + ,pos[x]);
} int query1(int o,int l,int r,int v)
{
if (l == r)
return sum1[o];
pushdown(o);
int mid = (l + r) >> ;
if (v <= mid)
return query1(o * ,l,mid,v);
else
return query1(o * + ,mid + ,r,v);
} int query2(int o,int l,int r,int v)
{
if (l == r)
return sum2[o];
pushdown(o);
int mid = (l + r) >> ;
if (v <= mid)
return query2(o * ,l,mid,v);
else
return query2(o * + ,mid + ,r,v);
} int main()
{
scanf("%d%d",&n,&m);
for (int i = ; i <= m; i++)
{
int x,y;
scanf("%d%d",&x,&y);
e[i].x = x;
e[i].y = y;
add(x,y);
add(y,x);
}
for (int i = ; i <= n; i++)
if (!h[i])
dfs1(i,),dfs2(i,i);
build(,,n);
for (int i = ; i <= m; i++)
{
if (!vis[i])
{
int x = e[i].x,y = e[i].y;
if (abs(h[x] - h[y]) % == ) //偶环
{
cntb++;
change(x,y,-);
}
else
{
cnta++;
flag = i;
change(x,y,);
}
}
}
if (cnta == )
{
for (int i = ; i <= m; i++)
anss[++ans] = i;
}
else
{
if (cnta == )
anss[++ans] = flag;
for (int i = ; i <= n; i++)
{
if (query1(,,n,pos[i]) == cnta && query2(,,n,pos[i]) == )
anss[++ans] = bianhao[i];
}
sort(anss + ,anss + + ans);
}
printf("%d\n",ans);
for(int i = ; i <= ans; i++)
printf("%d ",anss[i]);
printf("\n"); return ;
}
    

Codeforces 19.E Fairy的更多相关文章

  1. codeforces 19 D. Points(线段树+set二分)

    题目链接:http://codeforces.com/contest/19/problem/D 题意:给出3种操作:1)添加点(x,y),2)删除点(x,y),3)查询离(x,y)最近的右上方的点. ...

  2. Codeforces Round #404 (Div. 2) C. Anton and Fairy Tale 二分

    C. Anton and Fairy Tale 题目连接: http://codeforces.com/contest/785/problem/C Description Anton likes to ...

  3. Codeforces Gym100735 I.Yet another A + B-Java大数 (KTU Programming Camp (Day 1) Lithuania, Birˇstonas, August 19, 2015)

    I.Yet another A + B You are given three numbers. Is there a way to replace variables A, B and C with ...

  4. Codeforces Gym100735 G.LCS Revised (KTU Programming Camp (Day 1) Lithuania, Birˇstonas, August 19, 2015)

    G.LCS Revised   The longest common subsequence is a well known DP problem: given two strings A and B ...

  5. Codeforces Gym100735 E.Restore (KTU Programming Camp (Day 1) Lithuania, Birˇstonas, August 19, 2015)

    E - Restore Given a matrix A of size N * N. The rows are numbered from 0 to N-1, the columns are num ...

  6. Educational Codeforces Round 19 A, B, C, E(xjb)

    题目链接:http://codeforces.com/contest/797 A题 题意:给出两个数n, k,问能不能将n分解成k个因子相乘的形式,不能输出-1,能则输出其因子: 思路:将n质因分解, ...

  7. 【19.77%】【codeforces 570D】Tree Requests

    time limit per test2 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...

  8. 【19.46%】【codeforces 551B】ZgukistringZ

    time limit per test2 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...

  9. 【codeforces 785C】Anton and Fairy Tale

    [题目链接]:http://codeforces.com/contest/785/problem/C [题意] 容量为n的谷仓,每一天都会有m个谷子入仓(满了就视为m);第i天 会有i只鸟叼走i个谷子 ...

随机推荐

  1. 在进行分布式框架搭建的过程中,出现问题advised by org.springframework.transaction.interceptor.TransactionInterceptor.invoke(org.aopalliance.intercept.MethodInvocation)?

    今天在进行宜立方商城,进行文件配置的时间,遇到如下的问题,问题是:advised by org.springframework.transaction.interceptor.TransactionI ...

  2. Python:Python的运行过程

    1.Python是什么 和Java以及c#一样,Python也是一门基于虚拟机的语言.熟悉Java开发的人在命令行执行一个Java程序的过程通常如下: javac hello.java java he ...

  3. 线段树---poj2528 Mayor’s posters【成段替换|离散化】

    poj2528 Mayor's posters 题意:在墙上贴海报,海报可以互相覆盖,问最后可以看见几张海报 思路:这题数据范围很大,直接搞超时+超内存,需要离散化: 离散化简单的来说就是只取我们需要 ...

  4. 人生的第一篇blog

    开始写博客了,人生第一篇博客啊,要写些什么呢?想想也没有什么头绪,随便写写吧. 这学期要使用代码管理工具了,要写团队项目了.一直以来都是自己一个人在默默编程,没有过合作经历.对于代码的管理也只是一直在 ...

  5. 第九章 Mysql函数

    简介 数学函数:处理数字 字符串函数:处理字符串 日期和时间函数:处理日期和时间,获取时间 条件判断函数:控制条件选择 系统信息函数:获取MySQL系统信息,包括数据库名称,当前用户名和数据库版本 加 ...

  6. 第二周:Scrum Meeting

    一.站立会议 顾名思义,站立会议即是在敏捷开发的冲刺阶段,为了更好地平衡团队成员的“交流”和“集中注意力”之间的矛盾.通过每日例会,即团队成员通过面对面的交流,并且其中大多数团队成员站着对会议进行讨论 ...

  7. maven 添加tomcat依赖

    https://my.oschina.net/angel243/blog/178554

  8. Kettle 使用Json输入

    import java.math.BigDecimal; private static final String JD="jd"; private static final Str ...

  9. 利用Docker安装Web前端性能测试工具Sitespeed.io

    目录结构 一.Sitespeed.io概述 1.Sitespeed.io简介 2.Sitespeed.io使用场景 二.Sitespeed.io的安装和使用 1.安装Sitespeed.io 2.连接 ...

  10. VSVC2010中常用的C++11特性

    static_assert 声明 static_assert 声明在编译时测试软件断言,这与在运行时进行测试的其他断言机制不同. 如果断言失败,则编译也将失败,且系统将发出指定的错误消息. const ...