题目描述

你有一个无向连通图,边的总数为偶数。
设图中有k个奇点(度数为奇数的点),你需要把它们配成k/2个点对(显然k被2整除)。对于每个点对(u,v),你需要用一条长度为偶数(假设每条边长度为1)的路径将u和v连接。每条路径允许经过重复的点,但不允许经过重复的边。这k/2条路径之间也不能有重复的边。

输入

第一行有两个整数n,m(2<=n,m<=250000),分别表示点数、边数,m为偶数。
接下来m行,每行两个整数a,b(1<=a,b<=n,a≠b),表示a,b间连有一条边。不存在重边。保证奇点的数目不为零。

输出

如果你认为无解就输出NIE。
设图中有k个奇点,则输出由k/2部分组成,每个部分包含两行:第一行为u,v,l,表示连接的两个点,及路径长度。第二行为空格隔开的l个整数,表示u到v的路径。边按照输入顺序从1到m编号。
若有多组答案,任意输出其中一个。

样例输入

6 8
1 2
2 3
3 4
4 5
5 6
6 1
1 4
2 5

样例输出

样例输出:
1 5 2
6 5
2 4 2
8 4
另一种合法输出:
1 5 6
1 2 3 7 6 5
2 4 2
8 4
 

根据题目给出的路径没有重边的要求,我们可以尝试找出原图的一个欧拉回路,这样就能满足路径没有重复边的条件了。要使原图有欧拉回路就要保证没有奇度数点,我们新建一个点$x$,将它连边向所有奇度数点,这样就能保证原图有欧拉回路了。但这样还是不能保证每个路径都有偶数条边,我们将原图拆点建新图将一个点$u$拆成$u_{A},u_{B}$两个点,将原图边$(u,v)$变为$(u_{A},v_{B})$或$(u_{B},v_{A})$(具体如何连下面再说)。将所有原图奇度数点$u$在新图中连边$(u_{A},x)$,这样就保证了新图中一条边的两端点下标不同(即一定有一个$A$一个$B$),而任意一对$A$点之间的任意路径都有偶数条边。对于原图的边我们找出原图的任意一棵生成树,对于不在生成树上的边随便连上述哪种都行,然后我们规定一个生成树的根,从叶子结点往上倒推出每个点连向父亲的边是$(u_{A},v_{B})$还是$(u_{B},v_{A})$。因为原图边数为偶数,所有这样连完显然是可以保证每个点度数为偶数。因为只需要配对原图奇度数的点,也就是新图中与$x$相连的点,我们从$x$开始$dfs$,并记录沿途边的编号并打上访问标记,到一个点时只要有没标记的出边就可以走下去,当再一次走到$x$时就说明匹配了一对点。这样不断走下去,当$x$的出边都被标记时就说明所有点对都匹配完毕。这也就是说实际上我们并不需要求出新图的一个欧拉回路。有一点需要注意的是因为$dfs$找路径时走过的边就不会再走了,而每个点可能会被遍历许多次,所以,每次走完的边都要在链式前向星上删除掉,否则会被卡。

这里有两个版本供大家选择

#include<set>
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<cstdio>
#include<bitset>
#include<vector>
#include<cstring>
#include<iostream>
#include<algorithm>
#define ll long long
#define pr pair<int,int>
using namespace std;
char *p1,*p2,buf[100000];
#define nc() (p1==p2&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++)
int read() {int x=0; char c=nc(); while(c<48) c=nc(); while(c>47) x=(((x<<2)+x)<<1)+(c^48),c=nc(); return x;}
vector<pr>v[250010];
int vis[500010];
int to[1000010];
int head[500010];
int next[1000010];
int q[500010];
int num[10000010];
int cnt;
int S,T;
int n,m;
int x,y;
int d[250010];
int dep[250010];
int r[250010];
int tot=1;
void add(int x,int y,int id)
{
next[++tot]=head[x];
head[x]=tot;
to[tot]=y;
num[tot]=id;
}
void dfs(int x,int fa,int num)
{
dep[x]=dep[fa]+1;
int len=v[x].size();
for(int i=0;i<len;i++)
{
int to=v[x][i].first;
if(!dep[to])
{
dfs(to,x,v[x][i].second);
}
else if(dep[to]>dep[x])
{
r[x]^=1;
add(x<<1,to<<1|1,v[x][i].second);
add(to<<1|1,x<<1,v[x][i].second);
}
}
if(r[x])
{
r[x]^=1;
add(x<<1,fa<<1|1,num);
add(fa<<1|1,x<<1,num);
}
else
{
r[fa]^=1;
add(x<<1|1,fa<<1,num);
add(fa<<1,x<<1|1,num);
}
}
void dfs2(int x)
{
if(x==1)
{
if(cnt)
{
printf("%d %d %d\n",S,T,cnt);
for(int i=1;i<=cnt;i++)
{
printf("%d ",q[i]);
}
printf("\n");
}
cnt=S=T=0;
}
else
{
T=x>>1;
if(!S)
{
S=x>>1;
}
}
while(1)
{
int i=head[x];
if(!i)
{
break;
}
head[x]=next[i];
if(vis[i>>1])
{
continue;
}
vis[i>>1]=1;
if(num[i])
{
q[++cnt]=num[i];
}
dfs2(to[i]);
}
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=m;i++)
{
scanf("%d%d",&x,&y);
v[x].push_back(make_pair(y,i));
v[y].push_back(make_pair(x,i));
d[x]++,d[y]++;
}
for(int i=1;i<=n;i++)
{
r[i]=d[i]&1;
if(d[i]&1)
{
add(1,i<<1,0);
add(i<<1,1,0);
}
}
dfs(1,0,0);
dfs2(1);
}
#include<set>
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<cstdio>
#include<bitset>
#include<vector>
#include<cstring>
#include<iostream>
#include<algorithm>
#define pr pair<int,int>
using namespace std;
vector<pr>v[500010];
int vis[500010];
int to[1000010];
int head[500010];
int next[1000010];
int q[500010];
int num[10000010];
int cnt;
int S,T;
int n,m;
int x,y;
int d[500010];
int dep[500010];
int r[500010];
int f[500010];
int tot=1;
int find(int x)
{
if(f[x]==x)
{
return x;
}
return f[x]=find(f[x]);
}
void add(int x,int y,int id)
{
next[++tot]=head[x];
head[x]=tot;
to[tot]=y;
num[tot]=id;
r[x]++;
}
void dfs(int x,int fa,int num)
{
int len=v[x].size();
for(int i=0;i<len;i++)
{
if(v[x][i].first!=fa)
{
dfs(v[x][i].first,x,v[x][i].second);
}
}
if(fa)
{
if(r[x]&1)
{
add(x,fa+n,num);
add(fa+n,x,num);
}
else
{
add(x+n,fa,num);
add(fa,x+n,num);
}
}
}
void dfs2(int x)
{
while(1)
{
int i=head[x];
if(i==-1)
{
break;
}
head[x]=next[i];
if(vis[i>>1])
{
continue;
}
vis[i>>1]=1;
dfs2(to[i]);
q[++cnt]=num[i];
}
}
int main()
{
scanf("%d%d",&n,&m);
memset(head,-1,sizeof(head));
for(int i=1;i<=n;i++)
{
f[i]=i;
}
for(int i=1;i<=m;i++)
{
scanf("%d%d",&x,&y);
d[x]++,d[y]++;
int fx=find(x);
int fy=find(y);
if(fx!=fy)
{
f[fx]=fy;
v[x].push_back(make_pair(y,i));
v[y].push_back(make_pair(x,i));
}
else
{
add(x,y+n,i);
add(y+n,x,i);
}
}
for(int i=1;i<=n;i++)
{
if(d[i]&1)
{
add(0,i,m+i);
add(i,0,m+i);
}
}
dfs(1,0,0);
dfs2(0);
while(cnt)
{
S=q[cnt--]-m;
int i=cnt;
while(q[i]<=m&&i>1)i--;
T=q[i]-m;
printf("%d %d %d\n",S,T,cnt-i);
for(int j=cnt;j>i;j--)
{
printf("%d ",q[j]);
}
printf("\n");
cnt=i-1;
}
}

BZOJ3724PA2014Final Krolestwo——欧拉回路+构造的更多相关文章

  1. BZOJ3724 PA2014Final Krolestwo(欧拉回路+构造)

    如果没有长度为偶数的限制,新建一个点向所有奇点连边,跑欧拉回路即可,显然此时一定存在欧拉回路,因为所有点度数都为偶数. 考虑长度为偶数的限制,将每个点拆成两个点放进一个二分图里,那么每条原图中的边在二 ...

  2. hdu 4850 字符串构造---欧拉回路构造序列 递归+非递归实现

    http://acm.hdu.edu.cn/showproblem.php? pid=4850 题意:构造长度为n的字符序列.使得>=4的子串仅仅出现一次 事实上最长仅仅能构造出来26^4+4- ...

  3. CF36E Two Paths (欧拉回路+构造)

    题面传送门 题目大意:给你一张可能有重边的不保证联通的无向图,现在要在这个图上找出两条路径,恰好能覆盖所有边一次,根据边的编号输出方案,无解输出-1 一道很不错的欧拉路径变形题 首先要知道关于欧拉路径 ...

  4. NOI前训练日记

    向别人学习一波,记点流水帐.17.5.29开坑. 5.29 早晨看了道据说是树状数组优化DP的题(hdu5542),然后脑补了一个复杂度500^3的meet in the middle.然后死T... ...

  5. IOI 2020 集训队作业胡扯

    首先安慰自己:做的没集训队快很正常-- 很正常-- 做不完也很正常-- 很正常-- 全都不会做也很正常-- 很正常-- 表格 试题一 完成情况 试题二 完成情况 试题三 完成情况 cf549E cf6 ...

  6. 2015多校.MZL's endless loop(欧拉回路的机智应用 || 构造)

    MZL's endless loop Time Limit: 3000/1500 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Oth ...

  7. CF527E Data Center Drama(构造+欧拉回路)

    题目链接 大意: 给你一个无向图. 要求加最少的边,然后给这些无向图的边定向,使得每一个点的出入度都是偶数. 输出定向后的边数和边集. n<=10^5 m<=2*10^5 很巧妙的构造题- ...

  8. UVa 12118 nspector's Dilemma (构造+DFS+欧拉回路)

    题意:给定n个点,e条边和每条边的长度t,每两个点之间都有路相连,让你求一条最短的路经过这e条边. 析:刚开始想到要判连通,然后把相应的几块加起来,但是,第二个样例就不过,后来一想,那么有欧拉回路的还 ...

  9. CF429E Points and Segments 构造、欧拉回路

    传送门 如果把一条线段\([l,r]\)看成一条无向边\((l,r+1)\),从\(l\)走到\(r+1\)表示线段\([l,r]\)染成红色,从\(r+1\)走到\(l\)表示线段\([l,r]\) ...

随机推荐

  1. Java系统高并发之Redis后端缓存优化

    一:前端优化 暴露接口,按钮防重复(点击一次按钮后就变成禁用,禁止重复提交) 采用CDN存储静态化的页面和一些静态资源(css,js等) 二:Redis后端缓存优化 Redis 是完全开源免费的,遵守 ...

  2. [翻译] .NET Standard 2.1 公布

    [翻译] .NET Standard 2.1 公布 原文: Announcing .NET Standard 2.1 校对: Cloud 自从大约一年前发布 .NET Standard 2.0以来,我 ...

  3. R实战 第十二篇:随机数

    由R生成的随机数实际上伪随机数,也就是说,随机数是由某种算法而不是真正的随机过程产生的,随机数生成器需要一个初始值来生成数字,该初始值叫做种子.通过把种子设置为特定的值,可以保证每次运行同一段代码时都 ...

  4. [C#]SQL Server Express LocalDb(SqlLocalDb)的一些体会

    真觉得自己的知识面还是比较窄,在此之前,居然还不知道SqlLocalDb. SqlLocalDb是啥?其实就是简化SQL Server的本地数据库,可以这样子说,SQL Server既可以作为远程,也 ...

  5. IIS配置Url重写实现http自动跳转https的重定向方法(100%解决)

    引言 本文推荐阅读地址:https://www.52abp.com/BlogDetails/10008 这种文章网上可以说一搜一大把,但是我为什么还要写呢,因为一搜一把没把我气死,都是东抄西挪的东西, ...

  6. Jmeter(三十六)_运行过程中改变负载

    顾名思义,jmeter在做性能测试时,可以在不停止脚本的情况下修改负载压力,达到期望的测试效果.我们将通过Constant Throughput Timer(吞吐量计时器)和Beanshell服务器来 ...

  7. 微信小程序学习笔记以及VUE比较

    之前只是注册了一下微信小程序AppID,随便玩了玩HelloWorld!(项目起手式),但是最近看微信小程序/小游戏,崛起之势不可阻挡.小程序我来了!(果然,一入前端深似海啊啊啊啊啊~) 编辑器: S ...

  8. 使用 Markdown编辑

    作用: 学习笔记,整理日志, 发布日记,杂文,所见所想 撰写发布技术文稿(代码支持) 撰写发布学术论文(LaTeX 公式支持) sublime text3插件 输入 Shift + Ctrl + P, ...

  9. 16-使用Selenium模拟浏览器抓取淘宝商品美食信息

    淘宝由于含有很多请求参数和加密参数,如果直接分析ajax会非常繁琐,selenium自动化测试工具可以驱动浏览器自动完成一些操作,如模拟点击.输入.下拉等,这样我们只需要关心操作而不需要关心后台发生了 ...

  10. 解决远程连接mysql很慢的方法(网络正常)

    最近用mysql命令行或者JDBC远程连接mysql速度很慢,而且远大于ping时间.上网搜了一下,解决方案如下: 在/etc/mysql/my.cnf文件的[mysqld]部分加入:skip-nam ...