【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. 【OpenCV】透视变换矫正

    演示结果参考: 功能实现:运行程序,会显示图片的尺寸,按回车键后,依次点击需矫正的图片的左上.右上.左下.右下角,并能显示其坐标,结果弹出矫正后的图片,如图上的PIC2对话框.可以继续选择图片四个点进 ...

  2. 复习静态页面polo-360

    1.ps快捷键 ctrl+1 恢复到100% ctrl+0 适应屏幕大小 ctrl+r 显示标尺 辅助线的利用 矩形框--图像--裁剪:文件存储为web所用格式,注意选格式. 1个像素的平铺 雪碧图的 ...

  3. 2、Android-UI(布局待完成)

    2.3.布局 实现界面的整齐摆放各种控件需要使用布局来完成 布局是一种可用于放置很多控件的容器 可以按照一定的规律调整内部的控件位置 布局的内部不仅可以放置控件还可以放置布局 1.线性布局 Linea ...

  4. Golang测试包

    Golang测试包 golang自带了测试包(testing),直接可以进行单元测试.性能分析.输出结果验证等.简单看着官方文档试了试,总结一下: 目录结构和命令 使用golang的测试包,需要遵循简 ...

  5. 使用JedisCluster出现异常:java.lang.NumberFormatException

    在使用JedisCluster进行测试时出现如下异常: java.lang.NumberFormatException: For input string: "7004@17004" ...

  6. Go语言之旅:基本类型

    原文地址:https://learn-linux.readthedocs.io 欢迎关注我们的公众号:小菜学编程 (coding-fan) Go 内置了以下基本类型: 布尔 bool 字符串 stri ...

  7. [iOS]为git设置代理

    查看本地git配置信息 git config --global -e 查看自己***的代理地址和端口信息 为git添加代理 git config --global http.proxy https:/ ...

  8. S2-01

    类: 类是对象的虚例,类是对象的集合,类是对象的模板. 对象是类的实例.(可以 0到多个.) .NET Framwork vs版本 .NET Framwork 1.0 /1.1 vs2003 .NET ...

  9. shell习题第9题:sed的常用用法

    [题目要求] 把一个文本文档的前5行中包含字母的行删除掉,同时把6到10行中的全部字母删除掉. [核心要点] sed命令 [脚本] .txt |sed '/[a-zA-Z]/d' .txt |sed ...

  10. nginx详解反向代理、负载均衡、LNMP架构上线动态网站

    简介 Nginx是俄罗斯人编写的十分轻量级的HTTP服务器,Nginx,它的发音为“engine X”,是一个高性能的HTTP和反向代理服务器,同时也是一个IMAP/POP3/SMTP 代理服务器.N ...