【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. antlr-2.7.6.jar的作用

    项目中没有添加antlr-2.7.6.jar,hibernate不会执行hql语句 并且会报NoClassDefFoundError: antlr/ANTLRException错误

  2. 【51nod 1514】 美妙的序列

    题目 我们发现我们得正难则反 还是设\(f_i\)表示长度为\(i\)的序列个数 考虑容斥 \[f_i=i!-\sum_{j=1}^{i-1}f_j(i-j)!\] \(i!\)显然是总方案数,我们减 ...

  3. Django中模型(二)

    Django中模型(二) 三.定义模型 1.模型.属性.表.字段间的关系: 一个模型类在数据库中对应一张表:在模型类中定义的属性,对应该模型对照表中的字段. 2.定义属性 A.概述 ·django根据 ...

  4. python自动化测试框架学习

    今天发现python有多个框架可以用于自动化测试方面,下面整理了下splinter和urllib2框架,对于pywinauto框架和ruby框架先记录下以后需要用到再学习. python有个splin ...

  5. java中NULL与" "的区别

    null是没有地址""是有地址但是里面的内容是空的 null和""的区别 问题一: null和""的区别 String s=null; st ...

  6. Java clone() 浅克隆与深度克隆

    内容转自:http://www.blogjava.net/orangelizq/archive/2007/10/17/153573.html 现在Clone已经不是一个新鲜词语了,伴随着“多莉”的产生 ...

  7. sharepoint 配置个人网站容量

    we have a SharePoint 2013 Standard edition implementation and 80 users. We are now starting to use M ...

  8. emmet完整收录(html,css)

    emmet官网 https://emmet.io/ 语法篇 Child: > nav>ul>li <nav> <ul> <li></li&g ...

  9. ios开发遇到的问题

    运行后界面空白,Xcode跳转到APPDelegate.swift文件提示如下 第一种可能原因: 做输出口后在代码中重新命名了输出口 解决方法: 右键控件关闭输出口的连接,变回+号,将它重新连到代码的 ...

  10. typeof 和 instanceof 区别

    typeof操作符返回一个字符串,表示未经计算的操作数的类型. 可能返回值有:"undefined"."object"."boolean". ...