[欧拉路]CF1152E Neko and Flashback
题意:对于长为n的序列c和长为n - 1的排列p,我们可以按照如下方法得到长为n - 1的序列a,b,a',b'。
ai = min(ci, ci+1),bi = max(ci, ci+1)
a'i = ap[i],b'i = bp[i]。
现在给定a'和b',求一个合法的c或者无解。
解:仔细分析性质,发现在a和b中,c除了开头和结尾会出现1次之外,每个数都会出现两次,且相邻。
我们可以把c的开头找出来,然后根据开头确定c2,然后确定c3...最后到cn。
注意到这些数可能有重复的,于是我们要试图在中间插入一段。我一开始想的是链表后来发现很难写...
仔细分析,如果把a'和b'的每个位置当成边,数字当成点,就是求欧拉路。然后就没了......
关于欧拉路:就暴力DFS,把每条边都访问一次。回溯的时候把这条边入栈/把y入栈。
#include <bits/stdc++.h>
const int N = ;
struct Edge {
int nex, v, id, pre;
}edge[N << ]; int tp = ;
int X[N], xx, a[N], b[N], cnt[N], e[N], stk[N], top, deg[N];
bool vis[N];
inline void erase(int x, int i) {
int nex = edge[i].nex, pre = edge[i].pre;
if(e[x] == i && !nex) {
e[x] = ;
}
else if(e[x] == i) {
e[x] = nex;
edge[nex].pre = ;
return;
}
else if(!nex) {
edge[pre].nex = ;
}
else {
edge[nex].pre = pre;
edge[pre].nex = nex;
}
edge[i].nex = edge[i].pre = ;
return;
}
inline void add(int x, int y, int z) {
edge[++tp].v = y;
edge[tp].id = z;
edge[tp].nex = e[x];
edge[e[x]].pre = tp;
e[x] = tp;
return;
}
void DFS(int x) {
for(int i = e[x]; i; i = edge[i].nex) {
erase(x, i);
int y = edge[i].v;
if(vis[edge[i].id]) {
continue;
}
vis[edge[i].id] = ;
DFS(y);
stk[++top] = y;
}
return;
}
int main() {
int n;
scanf("%d", &n);
for(int i = ; i < n; i++) {
scanf("%d", &a[i]);
X[++xx] = a[i];
}
for(int j = ; j < n; j++) {
scanf("%d", &b[j]);
X[++xx] = b[j];
if(b[j] < a[j]) {
puts("-1");
return ;
}
}
std::sort(X + , X + xx + );
xx = std::unique(X + , X + xx + ) - X - ;
for(int i = ; i < n; i++) {
a[i] = std::lower_bound(X + , X + xx + , a[i]) - X;
b[i] = std::lower_bound(X + , X + xx + , b[i]) - X;
add(a[i], b[i], i);
add(b[i], a[i], i);
deg[a[i]]++;
deg[b[i]]++;
}
int s = , pos = ;
for(int i = ; i <= xx; i++) {
if(deg[i] & ) {
s++;
pos = i;
}
}
if(s != && s != ) {
puts("-1");
return ;
}
DFS(pos);
stk[++top] = pos;
if(top != n) {
puts("-1");
return ;
}
for(int i = top; i >= ; i--) {
printf("%d ", X[stk[i]]);
}
return ;
}
AC代码
注意复杂度,删边......
#include <bits/stdc++.h>
const int N = ;
struct Edge {
int nex, v, id;
}edge[N << ]; int tp = ;
int X[N], xx, a[N], b[N], cnt[N], e[N], stk[N], top, deg[N];
bool vis[N];
inline void add(int x, int y, int z) {
edge[++tp].v = y;
edge[tp].id = z;
edge[tp].nex = e[x];
e[x] = tp;
return;
}
void DFS(int x) {
for(int i = e[x]; i; i = e[x]) {
e[x] = edge[i].nex;
int y = edge[i].v;
if(vis[edge[i].id]) {
continue;
}
vis[edge[i].id] = ;
DFS(y);
stk[++top] = y;
}
return;
}
int main() {
int n;
scanf("%d", &n);
for(int i = ; i < n; i++) {
scanf("%d", &a[i]);
X[++xx] = a[i];
}
for(int j = ; j < n; j++) {
scanf("%d", &b[j]);
X[++xx] = b[j];
if(b[j] < a[j]) {
puts("-1");
return ;
}
}
std::sort(X + , X + xx + );
xx = std::unique(X + , X + xx + ) - X - ;
for(int i = ; i < n; i++) {
a[i] = std::lower_bound(X + , X + xx + , a[i]) - X;
b[i] = std::lower_bound(X + , X + xx + , b[i]) - X;
add(a[i], b[i], i);
add(b[i], a[i], i);
deg[a[i]]++;
deg[b[i]]++;
}
int s = , pos = ;
for(int i = ; i <= xx; i++) {
if(deg[i] & ) {
s++;
pos = i;
}
}
if(s != && s != ) {
puts("-1");
return ;
}
DFS(pos);
stk[++top] = pos;
if(top != n) {
puts("-1");
return ;
}
for(int i = top; i >= ; i--) {
printf("%d ", X[stk[i]]);
}
return ;
}
另一种删边方式
[欧拉路]CF1152E Neko and Flashback的更多相关文章
- 洛谷P1341 无序字母对[无向图欧拉路]
题目描述 给定n个各不相同的无序字母对(区分大小写,无序即字母对中的两个字母可以位置颠倒).请构造一个有n+1个字母的字符串使得每个字母对都在这个字符串中出现. 输入输出格式 输入格式: 第一行输入一 ...
- POJ1386Play on Words[有向图欧拉路]
Play on Words Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 11846 Accepted: 4050 De ...
- hdu1161 欧拉路
欧拉路径是指能从一个点出发能够“一笔画”完整张图的路径:(每条边只经过一次而不是点) 在无向图中:如果每个点的度都为偶数 那么这个图是欧拉回路:如果最多有2个奇数点,那么出发点和到达点必定为该2点,那 ...
- UVA10054The Necklace (打印欧拉路)
题目链接 题意:一种由彩色珠子组成的项链.每个珠子的两半由不同的颜色组成.相邻的两个珠子在接触的地方颜色相同.现在有一些零碎的珠子,需要确定他们是否可以复原成完整的项链 分析:之前也没往欧拉路上面想, ...
- 洛谷 P1341 无序字母对 Label:欧拉路 一笔画
题目描述 给定n个各不相同的无序字母对(区分大小写,无序即字母对中的两个字母可以位置颠倒).请构造一个有n+1个字母的字符串使得每个字母对都在这个字符串中出现. 输入输出格式 输入格式: 第一行输入一 ...
- POJ 1637 Sightseeing tour (混合图欧拉路判定)
Sightseeing tour Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 6986 Accepted: 2901 ...
- hihocoder 1181 欧拉路.二
传送门:欧拉路·二 #1181 : 欧拉路·二 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 在上一回中小Hi和小Ho控制着主角收集了分散在各个木桥上的道具,这些道具其 ...
- hiho48 : 欧拉路·一
时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho最近在玩一个解密类的游戏,他们需要控制角色在一片原始丛林里面探险,收集道具,并找到最后的宝藏.现在他们控制的 ...
- hdu5883 The Best Path(欧拉路)
题目链接:hdu5883 The Best Path 比赛第一遍做的时候没有考虑回路要枚举起点的情况导致WA了一发orz 节点 i 的贡献为((du[i] / 2) % 2)* a[i] 欧拉回路的起 ...
随机推荐
- 【JZOJ6388】小w的作业
description analysis 二分一个角度,首先假设该弧度角\(\theta \in[{\pi \over 2},\pi]\),要找的直线斜率\(k\in(-∞,\tan\theta]\) ...
- Controller 获取前端数据
默认支持的类型 在controller的方法的形参中直接定义上面这些类型的参数,springmvc会自动绑定. HttpServletRequest对象 HttpServletResponse对象 H ...
- PouchContainer 开源版本及内部版本一致性实践
PouchContainer 开源版本及内部版本一致性实践 为什么要做内外版本一致 对外开源是提升影响力.共建生态的有力手段.在项目对外开源的过程中,首先是将可以开源的部分抽离出来,发布一个“开源版本 ...
- LUOGU P3161 [CQOI2012]模拟工厂 (贪心)
传送门 解题思路 贪心,首先因为\(n\)比较小,可以\(2^n\)枚举子集.然后判断的时候就每次看后面的如果用最大生产力生产能不能达成目标,解一个二次函数. 代码 #include<iostr ...
- 使用Windbg调试系统弹出的内存不可读错误
步骤: 1. 使用Windbg挂钩到崩溃的进程上面 2. 使用~*k列出所有线程 3. 搜索UnhandledExceptionFilter所在的线程 4. 使用~ns切换到上面崩溃所在的线程,n为线 ...
- sql 查询问题
在做数据导出时候,当某个表某字段含有单引号时候老是报错,所以要排除这种情况: sql查询某表某字段值带单引号情况 select 主键码 from 馆藏书目库 where 题名 like '%''%' ...
- Georgia and Bob
Georgia and Bob 给出一个严格递增的正整数数列\(\{a_i\}\),每一次操作可以对于其中任意一个数减去一个正整数,但仍然要保证数列的严格递增性,现在两名玩家轮流操作,不能操作的玩家判 ...
- 关于Cadence SPB 16.6 破解的一些关键步骤
网上搜索到的破解方法,一般来说流程都是没有什么大问题的.较详细的一个破解流程可参照: http://www.mr-wu.cn/cadence-orcad-allegro-spb-16-6-instal ...
- Oracle使用语句块之循环插入数据
1.业务要求: 将oracle表A的整表的数据一次性导入到表B中 , 以A_ID为外键关联. (*******如果开发环境和实际生产环境的数据一致,而且数据量比较小情况,可以直接手动添加数据; ** ...
- 报javax.servlet.ServletException: Servlet.init() for servlet [springmvc] threw exception的解决记录
1.异常详情: 2.异常分析: 从异常的详情中看出:companyService未找到,出现这种情况的愿意可能是companyServiceImpl类没有交给IOC容器管理,但是经过我已经在该类上打了 ...