【LG5022】[NOIP2018]旅行

题面

洛谷

题解

首先考虑一棵树的部分分怎么打

直接从根节点开始\(dfs\),依次选择编号最小的儿子即可

而此题是一个基环树

怎么办呢?

可以断掉环上的一条边,这样就变为一棵树了

再用上面的方法做即可

\(tips\) \(:\)

断环上的边,其实可以直接用\(tarjan\)把桥求出来

不是桥的就是环上的边

考场上的代码有点乱

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <vector>
using namespace std;
inline int gi() {
register int data = 0, w = 1;
register char ch = 0;
while (ch != '-' && !isdigit(ch)) ch = getchar();
if (ch == '-') w = -1, ch = getchar();
while (isdigit(ch)) data = data * 10 + ch - '0', ch = getchar();
return w * data;
}
#define MAX_N 5005
vector<int> G[MAX_N];
struct Graph { int to, next; } e[MAX_N << 1]; int fir[MAX_N], e_cnt;
void clearGraph(){ memset(fir, -1, sizeof(fir)); }
void Add_Edge(int u, int v) { e[e_cnt] = (Graph){v, fir[u]}; fir[u] = e_cnt++; }
struct Edge {
int to, id;
bool operator < (const Edge &rhs) const { return to < rhs.to; }
} ;
vector<Edge> rG[MAX_N];
int N, M;
namespace cpp1 {
void dfs(int x, int f) {
printf("%d ", x);
for (int i = 0, sz = rG[x].size(); i < sz; i++) {
int v = rG[x][i].to;
if (v == f) continue;
dfs(v, x);
}
}
void main() { dfs(1, 0); putchar('\n'); }
} namespace cpp2 {
int dfn[MAX_N], low[MAX_N], tim;
bool bridge[MAX_N << 1];
void tarjan(int x, int id) {
dfn[x] = low[x] = ++tim;
for (int i = fir[x]; ~i; i = e[i].next) {
int v = e[i].to;
if (!dfn[v]) {
tarjan(v, i), low[x] = min(low[x], low[v]);
if (low[v] > dfn[x]) bridge[i] = bridge[i ^ 1] = 1;
} else if (i != (id ^ 1)) low[x] = min(low[x], dfn[v]);
}
}
int ans[MAX_N], tmp[MAX_N], cnt;
bool used[MAX_N << 1];
void dfs(int x, int fa) {
tmp[++cnt] = x;
for (int i = 0, sz = rG[x].size(); i < sz; i++) {
int v = rG[x][i].to, id = rG[x][i].id;
if (v == fa || used[id]) continue;
dfs(v, x);
}
}
bool check() {
for (int i = 1; i <= N; i++) {
if (ans[i] > tmp[i]) return 1;
else if (ans[i] < tmp[i]) return 0;
}
return 0;
}
void main() {
tarjan(1, -1);
for (int i = 1; i <= N; i++) ans[i] = N + 1;
for (int i = 0; i < e_cnt; i += 2) {
if (bridge[i]) continue;
used[i] = used[i ^ 1] = 1; cnt = 0;
dfs(1, 0);
if (check()) for (int j = 1; j <= N; j++) ans[j] = tmp[j];
used[i] = used[i ^ 1] = 0;
}
for (int i = 1; i <= N; i++) printf("%d ", ans[i]);
putchar('\n');
}
}
bool used[MAX_N][MAX_N];
int main () {
N = gi(), M = gi();
for (int i = 1; i <= M; i++) {
int u = gi(), v = gi();
G[u].push_back(v), G[v].push_back(u);
}
for (int i = 1; i <= N; i++) sort(G[i].begin(), G[i].end());
clearGraph();
for (int i = 1; i <= N; i++)
for (int j = G[i].size() - 1; j >= 0; j--) {
int v = G[i][j]; if (used[i][v]) continue;
Add_Edge(i, v), Add_Edge(v, i);
used[i][v] = used[v][i] = 1;
}
for (int x = 1; x <= N; x++) {
for (int i = fir[x]; ~i; i = e[i].next) {
int v = e[i].to;
rG[x].push_back((Edge){v, i});
}
}
for (int i = 1; i <= N; i++) sort(rG[i].begin(), rG[i].end());
if (N - 1 == M) cpp1::main();
else cpp2::main();
return 0;
}

【LG5022】[NOIP2018]旅行的更多相关文章

  1. 竞赛题解 - NOIP2018 旅行

    \(\mathcal {NOIP2018} 旅行 - 竞赛题解\) 坑还得一层一层的填 填到Day2T1了 洛谷 P5022 题目 (以下copy自洛谷,有删减/修改 (●ˇ∀ˇ●)) 题目描述 小 ...

  2. NOIP2018 旅行 和 赛道修建

    填很久以前的坑. 旅行 给一棵 n 个点的基环树,求字典序最小的DFS序. n ≤ 5000 题解 O(n2) 做法非常显然,枚举断掉环上哪条边然后贪心即可.当然我去年的骚操作只能得88分. O(n ...

  3. [NOIP2018]:旅行(数据加强版)(基环树+搜索+乱搞)

    题目描述 小$Y$是一个爱好旅行的$OIer$.她来到$X$国,打算将各个城市都玩一遍.小$Y$了解到,$X$国的$n$个城市之间有$m$条双向道路.每条双向道路连接两个城市.不存在两条连接同一对城市 ...

  4. NOIP2018旅行

    这道题考场上的时候暴力写RE了,我果然很菜. 看了一篇大佬的的题解才明白 dalao的题解 但是解释很少哇,为了造福人类,在下发一篇详细一点的题解. 预处理:用vector把与每个点相连的点存起来,排 ...

  5. 【比赛】NOIP2018 旅行

    发现 \(m\) 只有两种取值,于是可做了 树的直接贪心 图的枚举环上的边去掉,然后做树的贪心,搜的时候剪一下枝吧 写得有点乱 #include<bits/stdc++.h> #defin ...

  6. luogu5022 [NOIp2018]旅行 (dfs)

    m=n-1的时候,就直接贪心地dfs就可以 m=n的话,就可以枚举删掉一条边,然后照着m=n-1做 $O(n^2)$大概能过 (然而我眼瞎看不到m<=n) #include<cstdio& ...

  7. [NOIP2018]旅行(数据加强版)(图论+基环树)

    数据范围多了2个0就是不一样,O(n^2)只能68分了.(其中60分是n=m+1和原题一样的做法送的),这题直接从NOIP难度变为NOI Plus难度了不说废话直接写题解:首先dfs一遍找到环,然后和 ...

  8. [NOIP2018]旅行

    嘟嘟嘟 鉴于一些知道的人所知道的,不知道的人所不知道的原因,我来发NOIPday2T1的题解了. \(O(n ^ 2)\)的做法自然很暴力,枚举断边断环为链就行了. 所以我是来讲\(O(nlogn)\ ...

  9. 【题解】NOIP2018 旅行

    题目戳我 \(\text{Solution:}\) 首先题目描述有一点不准确:回头是必须要走完一条路无路可走的时候才能返回. 对于树的情况:显然贪心做就完事了. 对于基环树的情况:对于一个\(n\)条 ...

随机推荐

  1. Codeforces Round #439 (Div. 2)【A、B、C、E】

    Codeforces Round #439 (Div. 2) codeforces 869 A. The Artful Expedient 看不透( #include<cstdio> in ...

  2. BZOJ 1001 狼抓兔子 平面图的最小割

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1001 题目大意: 见链接 思路: 求最小割,平面图的最小割等价于对偶图的最短路 直接建 ...

  3. P1395 会议

    题目描述 有一个村庄居住着n个村民,有n-1条路径使得这n个村民的家联通,每条路径的长度都为1.现在村长希望在某个村民家中召开一场会议,村长希望所有村民到会议地点的距离之和最小,那么村长应该要把会议地 ...

  4. Linux下安装PHP并在nginx服务器中进行配置的详细方法

    先介绍一下使用的环境:centos 7.4, PHP 7.0 , nginx 1.12 Linux系统版本可以通过命令:lsb_release -a 查看. 现在开始步入正题了! 1.  首先查看一下 ...

  5. 子查询 SQL

    SELECT *,(SELECT COUNT(*) FROM yd_order o WHERE FROM_UNIXTIME(o.`ctime`,'%Y-%m')='2016-06' AND o.uid ...

  6. LCG(linear congruential generator): 一种简单的随机数生成算法

    目录 LCG算法 python 实现 LCG算法 LCG(linear congruential generator)线性同余算法,是一个古老的产生随机数的算法.由以下参数组成: 参数 m a c X ...

  7. 用JavaScript中jQuery编写放大镜效果

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  8. Mac 10.8.5 上运行cgi

    搭配置搭了好久,花了近半天时间,有必要把过程记录下. 本文已同步到 icocoa.tk, 欢迎访问 Mountain Lion上Setting设置里已经取消了web share,必须要自己启动apac ...

  9. js中的冒泡排序

    <!-- 冒泡排序:把一组数列按照一定的顺序进行排列,从大到小,或者从小到大 -->          // 控制循环的轮数          arr = [3,14,55,2,1,4,5 ...

  10. JAVA WEB 前台实时监控后台程序运行

    基本思路: 1. 操作状态在类中以静态变量方式(或公共类存储公共变量方式,SESSION方式.COOKIE方式)存在 2. 前台采用AJAX方式激发后台进行业务逻辑操作,并实时更新操作状态信息 3.  ...