【欧拉回路+最小生成树】SD开车@山东2018省队一轮集训day1

PROBLEM

题目描述

作为钦钦草原最绿的男人,杨某针每天都要开车巡视钦钦草原一圈。

钦钦草原由 n 个城市组成,m 条双向道路连接着它们。经过第 i 条道路要花费的时间是\(2^i\)。

杨某针想要经过每条道路至少一次,在此基础上他想最小化他花费的时间。但作为 曾经 CTSC 的 Cu 选手,他并不能很快地计算出这个问题。所以他向你求助。

输入

输入第一行包含两个正整数n,m。

接下来m行,每行两个正整数\(a_i\),\(b_i\),表示第i条边连接点\(a_i\)和\(b_i\),它的权值为\(2^i\)。保证\(a_i\neq b_i\),不存在重边,且任意两个点之间可以互相到达。

输出

输出一行一个整数,表示答案对\(10^9\)+7取模的值。

样例输入

4 5

1 2

3 4

2 3

1 3

2 4

样例输出

70

提示

最优的路线应当为 1-2-3-4-2-3-1。

对于20%的数据,n,m≤20。

对于40%的数据,n,m≤2,000。

对于100%的数据,n≤400,000,m≤500,000。

SOLUTION

若存在一条从节点S出发的路径,恰好不重不漏地经过每条边一次(可以重复经过图中的节点),最终回到起点S,则称该路径为欧拉回路。存在欧拉回路的无向图被称为欧拉图

要经过每条道路至少一次,可以对比在欧拉回路中,每条边恰好经过一次。若最佳路线经过某一条边n+1次,可以看作在原图中加上了n次那条边。由原图G加上重复走的边得到G',G'一定是一个欧拉图,由欧拉图的性质,G'中每个点的度一定为偶数。

题目要求最小花费,也就是要加的边的权值和最小。若我分若干次去添加边,这里可以贪心假设,每次加的一个边集一定是某两个度为奇数的点的最短路径上的边,然后使得这两个点的度变为偶数。

而至于如何去求两个奇数度点的最短路,可以从边的权值上下手——第i条边的权值是\(2^i\)。用求最小生成树的Kruskal算法,我们从按编号(也就是权值)小到大枚举边,然后并查集维护,建出一棵最小生成树。因为\(2^n = 2^{n-1} + 2^{n-2} + ... + 2^1+ 2\) ,所以生成树上两点距离就是原图中两点最短距离(这里需要仔细思考)。

最后dfs遍历生成树,回溯时判断当前点的度数是否为奇数,如果是奇数,让答案再加上该点和它的父节点所连的边的权值,并更新两点的度数。

最终答案要加上原图中所有边的权值和。

CODE

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int MAXN = 4e5 + 5;
const int MAXE = 5e5+5;
const int INF = 0x3f3f3f3f;
const ll MOD = 1e9+7; int degree[MAXN];
int u[MAXE],v[MAXE]; struct edge{
int u,v,w,nex;
}ed[MAXN<<1]; int head[MAXN],tot; void addedge(int uu,int vv,int w){
tot++;
ed[tot].u = uu;
ed[tot].v = vv;
ed[tot].w = w;
ed[tot].nex = head[uu];
head[uu] = tot;
} int fa[MAXN]; int DjsGet(int x){
if(x==fa[x])return x;
return fa[x] = DjsGet(fa[x]);
} ll fastpow(ll a,ll n){
ll res = 1;
while(n){
if(n&1)res=res*a%MOD;
a = a*a%MOD;
n>>=1;
}
return res;
} ll ans; void dfs(int u,int p){
for(int i=head[u];i;i = ed[i].nex){
int v = ed[i].v;
if(i!=(p^1))
dfs(v,i);
}
if(degree[u]&1){
ans=(ans+fastpow(2,ed[p].w))%MOD;
degree[u]++;
degree[ed[p].u]++;
}
} int main() {
int n,m;
scanf("%d%d",&n,&m);
for(int i=1;i<=m;i++){
scanf("%d%d",&u[i],&v[i]);
degree[u[i]]++;
degree[v[i]]++;
}
tot++;
for(int i = 1;i<=n;i++)fa[i] = i;
for(int i=1;i<=m;i++){
int fx = DjsGet(u[i]),fy = DjsGet(v[i]);
if(fx!=fy){
fa[fx] = fy;
addedge(u[i],v[i],i);
addedge(v[i],u[i],i);
}
ans=(ans+fastpow(2,i))%MOD;
}
dfs(ed[2].u,0);
printf("%lld\n",ans);
return 0;
}

【欧拉回路+最小生成树】SD开车@山东2018省队一轮集训day1的更多相关文章

  1. LOJ 6060「2017 山东一轮集训 Day1 / SDWC2018 Day1」Set(线性基,贪心)

    LOJ 6060「2017 山东一轮集训 Day1 / SDWC2018 Day1」Set $ solution: $ 这一题的重点在于优先级问题,我们应该先保证总和最大,然后再保证某一个最小.于是我 ...

  2. SD 一轮集训 day1 carcar

    可以发现每条边只能选一次或者两次,并且最后每个点的度数(∑邻接边选的次数和)都是偶数(代表有欧拉回路). 然后根据题意列一个 n 行 m+1 列的01矩阵,每一行代表一个异或方程组(每个点的度数是偶数 ...

  3. LOJ #6060. 「2017 山东一轮集训 Day1 / SDWC2018 Day1」Set

    有趣的思博套路题,想到了基本上加上个对线性基的理解就可以过了 首先考虑到这个把数分成两半的分别异或的过程不会改变某一位上\(1\)的总个数 因此我们求出所有数的\(\operatorname{xor} ...

  4. LOJ.6060.[2017山东一轮集训Day1/SDWC2018Day1]Set(线性基)

    LOJ BZOJ 明明做过一道(最初思路)比较类似的题啊,怎么还是一点思路没有. 记所有元素的异或和为\(s\),那么\(x_1+x_2=x_1+x_1\ ^{\wedge}s\). \(s\)是确定 ...

  5. 【LOJ6060】【2017 山东一轮集训 Day1 / SDWC2018 Day1】Set 线性基

    题目大意 给出 \(n\) 个非负整数,将数划分成两个集合,记为一号集合和二号集合.\(x_1\) 为一号集合中所有数的异或和,\(x_2\) 为二号集合中所有数的异或和.在最大化 \(x_1 + x ...

  6. loj6102 「2017 山东二轮集训 Day1」第三题

    传送门:https://loj.ac/problem/6102 [题解] 贴一份zyz在知乎的回答吧 https://www.zhihu.com/question/61218881 其实是经典问题 # ...

  7. 【LOJ6060】「2017 山东一轮集训 Day1 / SDWC2018 Day1」Set(线性基)

    点此看题面 大致题意: 让你把\(n\)个数分成两部分,使得在两部分异或和之和最大的前提下,两个异或和中较小的那个尽量小.输出最优的较小异或和. 线性基 关于线性基,可以看一下这篇博客:线性基入门. ...

  8. loj6100 「2017 山东二轮集训 Day1」第一题

    传送门:https://loj.ac/problem/6100 [题解] 我们考虑维护从某个端点开始的最长满足条件的长度,如果知道了这个东西显然我们可以用主席树来对每个节点建棵关于右端点的权值线段树, ...

  9. SD 一轮集训 day1 lose

    神TM有是结论题,我讨厌结论题mmp. 杨氏矩阵了解一下(建议去维基百科). 反正就是推柿子,使劲推,最后写起来有一点小麻烦,但是在草稿纸(然鹅我木有啊)上思路清晰的话还是没问题的. #include ...

随机推荐

  1. DirectX11 With Windows SDK--09 纹理映射与采样器状态

    前言 在之前的DirectX SDK中,纹理的读取使用的是D3DX11CreateShaderResourceViewFromFile函数,现在在Windows SDK中已经没有这些函数,我们需要找到 ...

  2. Resources (being shared)

    论文下载求助论坛 数学杂志的模版 答辩PPT模版 发一篇文章的经历 数学期刊名称缩写 英文书籍下载 英文书籍下载 中文书籍下载 数学分析高等代数考研试题官方下载地址

  3. “Django用户认证系统”学习资料收集

    首推追梦人物——Django用户认证系统 待续……

  4. ES6走一波 数组的扩展

    Array flat 数组实例的扁平化方法(浏览器支持不佳) 建议使用 lodash的 flatten

  5. vue组件化的应用

    前言:vue组件化的应用涉及到vue-cli的内容,所以在应用之前是需要安装node和vue-cli的,具体如何安装我就不一一赘述了.可能一会儿我心情好的时候,可以去整理一下. 1.应用的内容:在一个 ...

  6. CF1119A Ilya and a Colorful Walk

    题目地址:CF1119A Ilya and a Colorful Walk \(O(n^2)\) 肯定过不掉 记 \(p_i\) 为从下标 \(1\) 开始连续出现 \(i\) 的个数 那么对于每一个 ...

  7. 51nod--1183 编辑距离(动态规划)

    题目: 1183 编辑距离 基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题 收藏 关注 编辑距离,又称Levenshtein距离(也叫做Edit Distance),是指 ...

  8. 【原创】数据库基础之Mysql(2)主从库配置

    一 安装 # wget -i -c http://dev.mysql.com/get/mysql57-community-release-el7-10.noarch.rpm# yum -y insta ...

  9. 小程序获取formid配置模板消息

    小程序无限获取formid,发送模板信息 1.发送模板信息需要条件:formid 2.formid产生环境:提交form表单产生,并且只有真机才能出现————安卓一个13位的时间戳(近期使用得时候,安 ...

  10. C语言学习及应用笔记之五:C语言typedef关键字及其使用

    在C语言中有一个typedef关键字,其用来定义用户自定义类型.当然,并不是真的创造了一种数据类型,而是给已有的或者符合型的以及复杂的数据类型取一个我们自己更容易理解的别名.总之,可以使用typede ...