@题目描述@

小 Y 是一个爱好旅行的 OIer。她来到 X 国,打算将各个城市都玩一遍。

小Y了解到, X国的 n 个城市之间有 m 条双向道路。每条双向道路连接两个城市。 不存在两条连接同一对城市的道路,也不存在一条连接一个城市和它本身的道路。并且, 从任意一个城市出发,通过这些道路都可以到达任意一个其他城市。小 Y 只能通过这些道路从一个城市前往另一个城市。

小 Y 的旅行方案是这样的:任意选定一个城市作为起点,然后从起点开始,每次可以选择一条与当前城市相连的道路,走向一个没有去过的城市,或者沿着第一次访问该城市时经过的道路后退到上一个城市。当小 Y 回到起点时,她可以选择结束这次旅行或 继续旅行。需要注意的是,小 Y 要求在旅行方案中,每个城市都被访问到。

为了让自己的旅行更有意义,小 Y 决定在每到达一个新的城市(包括起点)时,将 它的编号记录下来。她知道这样会形成一个长度为 n 的序列。她希望这个序列的字典序 最小,你能帮帮她吗?

输入

输入文件共 m + 1 行。第一行包含两个整数 n,m(m≤n),中间用一个空格分隔。

接下来 m 行,每行包含两个整数 u,v(1≤u,v≤n) ,表示编号为 u 和 v 的城市之间有一条道路,两个整数之间用一个空格分隔。

输出

输出文件包含一行,n 个整数,表示字典序最小的序列。相邻两个整数之间用一个空格分隔。

输入样例#1

6 5

1 3

2 3

2 5

3 4

4 6

输出样例#1

1 3 2 5 4 6

输入样例#2

6 6

1 3

2 3

2 5

3 4

4 5

4 6

输出样例#2

1 3 2 4 5 6

数据规模与约定

对于 100% 的数据和所有样例, 1≤n≤5000 且 m = n − 1 或 m = n。

@题解@

据说这道题有很神奇的做法……可以 O(n) …… orz

但是介于我实在太弱了,所以只能在此向大家介绍 O(n^2) 的一个简单算法。

不难发现题目给出的其实要么是树要么是基环树。

树的话只需要贪心地走它编号最小的儿子就可以实现字典序最小。

如果是基环树的话,我们需要转化成一棵树。

基环树转树最常见的肯定是删去某一条环上的边。

所以,如果是基环树,我们就枚举一条环上的边,将它打上标记。再 dfs 一遍(要求不经过这条被标记的边),求得一个答案。最后再所有的答案取一个最小值。

以 1 为起点显然比较优秀。

@代码@

因为是第一题所以比较轻松。

可能是因为害怕选手 D2 甚至无法 AC 一道题,然后怒而退役吧。

#include<queue>
#include<cstdio>
#include<vector>
#include<algorithm>
using namespace std;
const int MAXN = 5000;
struct node{
int num, to;
node(int _n, int _t):num(_n), to(_t){}
};
bool operator < (node a, node b) {
return a.to < b.to;
}
vector<node>G[MAXN + 5];
void addedge(int u, int v, int i) {
G[u].push_back(node(i, v));
G[v].push_back(node(i, u));
}
bool vis[MAXN + 5];
int res[MAXN + 5], ans[MAXN + 5], cnt;
void init(int n) {
for(int i=1;i<=n;i++)
vis[i] = false;
cnt = 0;
}
int tag[MAXN + 5], deg[MAXN + 5];
queue<int>que;
void find_circle(int n) {
for(int i=1;i<=n;i++)
if( deg[i] == 1 ) que.push(i);
while( !que.empty() ) {
int f = que.front(); que.pop();
vis[f] = true;
for(int i=0;i<G[f].size();i++) {
if( vis[G[f][i].to] ) continue;
tag[G[f][i].num] = 1;
deg[G[f][i].to]--;
if( deg[G[f][i].to] == 1 )
que.push(G[f][i].to);
}
}
}
void dfs(int x) {
res[++cnt] = x; vis[x] = true;
for(int i=0;i<G[x].size();i++)
if( !vis[G[x][i].to] && tag[G[x][i].num] != -1 )
dfs(G[x][i].to);
}
int main() {
int n, m;
scanf("%d%d", &n, &m);
for(int i=1;i<=m;i++) {
int u, v;
scanf("%d%d", &u, &v);
addedge(u, v, i); deg[u]++, deg[v]++;
}
for(int i=1;i<=n;i++)
sort(G[i].begin(), G[i].end());
if( n == m ) {
init(n); find_circle(n); ans[1] = -1;
for(int i=1;i<=m;i++) {
if( !tag[i] ) {
tag[i] = -1; init(n); dfs(1); tag[i] = 0;
bool flag = false;
if( ans[1] == -1 ) flag = true;
else {
for(int i=1;i<=cnt;i++) {
if( ans[i] > res[i] ) {
flag = true;
break;
}
else if( ans[i] < res[i] )
break;
}
}
if( flag ) {
for(int i=1;i<=cnt;i++)
ans[i] = res[i];
}
}
}
for(int i=1;i<n;i++)
printf("%d ", ans[i]);
printf("%d\n", ans[n]);
}
else {
init(n); dfs(1);
for(int i=1;i<cnt;i++)
printf("%d ", res[i]);
printf("%d\n", res[cnt]);
}
}

@NOIP2018 - D2T1@ 旅行的更多相关文章

  1. noip 2018 d2t1 旅行

    noip 2018 d2t1 旅行 (题目来自洛谷) 给定n个城市,m条双向道路的图, 不存在两条连接同一对城市的道路,也不存在一条连接一个城市和它本身的道路.并且, 从任意一个城市出发,通过这些道路 ...

  2. [NOIP2018 TG D2T1]旅行

    题目大意:$NOIP\;TG\;D2T1$ 题解:一棵树的很简单,第一个点一定是$1$,只需要对每个节点,找最小的没有访问过的节点访问即可,我写的是$O(n\log_2n)$. 考虑基环树的部分,一个 ...

  3. NOIp 2018 D2T1 旅行//未完成

    这个题没有认真读的话就会写下以下的DD代码 #include<bits/stdc++.h> #define N 5010 using namespace std; int n,m; int ...

  4. $Noip2018/Luogu5022$ 旅行

    $Luogu$ $Description$ 一个$n$个点,$m$条边的图.$m=n-1$或$m=n$.任意选取一点作为起始点,可以去往一个没去过的点,或者回到第一次到达这个点时来自的点.要求遍历整个 ...

  5. NOIP2018提高组题解

    D1T1:铺设道路 回忆NOIP2013D2T1 积木大赛,发现这两题唯一的区别就是一个是造山一个是填坑,而把填坑的操作反序就是造山,所以可以直接使用那道题的方法. 具体方法是,从左到右每次考虑新的一 ...

  6. NOIP2018 解题笔记

    D1T1 铺设道路 在场上并没有想到积木大赛这道原题. 差分之后可以把在$[l, r]$这段区间$ - 1$变成在$l$处$ - 1$,在$r + 1$处$ + 1$,然后最终目标是使$\forall ...

  7. BZOJ 4016 [FJOI2014]最短路径树问题 (贪心+点分治)

    题目大意:略 传送门 硬是把两个题拼到了一起= = $dijkstra$搜出单源最短路,然后$dfs$建树,如果$dis_{v}=dis_{u}+e.val$,说明这条边在最短路图内,然后像$NOIP ...

  8. [NOIp2018提高组]旅行

    [NOIp2018提高组]旅行: 题目大意: 一个\(n(n\le5000)\)个点,\(m(m\le n)\)条边的连通图.可以从任意一个点出发,前往任意一个相邻的未访问的结点,或沿着第一次来这个点 ...

  9. 【LG5022】[NOIP2018]旅行

    [LG5022][NOIP2018]旅行 题面 洛谷 题解 首先考虑一棵树的部分分怎么打 直接从根节点开始\(dfs\),依次选择编号最小的儿子即可 而此题是一个基环树 怎么办呢? 可以断掉环上的一条 ...

随机推荐

  1. day18 10.使用ThreadLocal来解决问题

    ThreadLocal是一个容器/集合,是一个Map集合.不管你跨多少层,只要你是同一个线程就可以取出来.Service和Dao是同一个线程.Service第一次调用JdbcUtils.getConn ...

  2. laravel-- 在laravel操作redis数据库的数据类型(string、哈希、无序集合、list链表、有序集合)

    安装redis和连接redis数据库 在controller头部引入 一.基本使用 public function RedisdDbOne() { // 清空Redis数据库 Redis::flush ...

  3. Python3 中 configparser 模块用法

    configparser 简介 configparser 是 Pyhton 标准库中用来解析配置文件的模块,并且内置方法和字典非常接近.Python2.x 中名为 ConfigParser,3.x 已 ...

  4. 利用CSS使footer固定在页面底部

    1.HTML基本结构 <!DOCTYPEhtml> <htmlxmlns="http://www.w3.org/1999/xhtml"> <headr ...

  5. HTML-DOM常用对象的用法(select/option/form/table)

    HTML DOM 常用对象: 它对常用HTML元素操作的简化. Select对象 它代表页面上的一个select元素,常用属性有: select.value ——当前选中项的value ,没有valu ...

  6. 数据库lib7第4题创建存储过程

    1. 传入2个字符串变量,其中,每个字符串是用分号(:)分隔的字串形式, 比如str1=’ab12;ab;cccc;tty’, str2=’1;6sf;8fffff;dd’, 注意,字符串是用户输入的 ...

  7. 剑指offer 1-6

    1. 二维数组中的查找 在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数.   分析 ...

  8. Spring_自动组件扫描和 基于注解配置bean

    自动组件扫描 启用Spring组件扫描功能. 使用@Component注释来表示这是类是一个自动扫描组件.  package com.tanlei.dao; import org.springfram ...

  9. 运行docker容器镜像2(指定容器启动时启动的脚本)

    docker中启动容器有以下两种情况. 第一种是通过 # docker run containerid 启动一个容器. 第二种是重新启动已经关闭的容器. # docker start containe ...

  10. chrome://inspect调试html页面空白,DOM无法加载的解决方案

    chrome://inspect调试html页面空白,DOM无法加载的解决方案 先描述一下问题 有一段时间没碰huilder hybird app 开发了,今天调试的时候 chrome://inspe ...