【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. keepalived.md

    配置文件说明 global_defs区域 global_defs { notification_email { acassen@firewall.loc failover@firewall.loc s ...

  2. CString char BSTR 转换

     关于字符集不一的历史原因,可以参考: UNICODE与ANSI的区别 以下是网上转载的资料.我将辅以自己的实例,说明并总结关系. 一.CString, int, string, char*之间的转换 ...

  3. BZOJ1042:[HAOI2008]硬币购物(DP,容斥)

    Description 硬币购物一共有4种硬币.面值分别为c1,c2,c3,c4.某人去商店买东西,去了tot次.每次带di枚ci硬币,买si的价值的东西.请问每次有多少种付款方法. Input 第一 ...

  4. Python常用库之二:Pandas

    Pandas是用于数据操纵和分析,建立在Numpy之上的.Pandas为Python带来了两种新的数据结构:Pandas Series和Pandas DataFrame,借助这两种数据结构,我们能够轻 ...

  5. Maven创建项目一些常见的问题

    1 .创建的项目中没有src/main/java.没有src/test/java 主要原因在于在创建项目的时候,使用的是系统自带的jdk,修改方法: 右键项目——Properties——javaBui ...

  6. myeclipse2014黑色主题风格设置

    http://jingyan.baidu.com/article/915fc41494db8451384b2043.html?st=2&os=0&bd_page_type=1& ...

  7. DBCP数据库连接池的简单使用

    0.DBCP简介      DBCP(DataBase connection pool)数据库连接池是 apache 上的一个Java连接池项目.DBCP通过连接池预先同数据库建立一些连接放在内存中( ...

  8. mysql 常用的时间日期函数小结

    本文主要是总结一些常用的在实际运用中常用的一些mysql时间日期以及转换的函数 1.now()  :返回当前日期和时间 select now(); //2018-04-21 09:19:21 2.cu ...

  9. iview-cli 项目、iView admin 代理与跨域问题解决方案

    iview-cli 项目.iView admin 跨域.代理问题解决方案 在webpack.dev.config.js文件中: 添加: devServer: { historyApiFallback: ...

  10. ACM1009:FatMouse' Trade

    Problem Description FatMouse prepared M pounds of cat food, ready to trade with the cats guarding th ...