UVA - 12263 Rankings 模拟(拓扑排序)
2024-08-26 09:59:38
题意:1~n这n个数,给你一个初始的顺序,再告诉你那两个数的大小关系发生了变化,求变化后的 顺序,不存在则输出IMPOSSIBLE
思路:这题很遗憾没在比赛的时候过掉,结束后加了一行就AC了。题目真的不难,我就是根据原顺序和变化得到任意两个数之间的大小关系。然后枚举变化后的这些数对,用构造法构造一个合法的序列,最后再和原顺序进行不重复的合并。两个数组,两个指针,合并的时候未发生变化的当前数若大于变化的当前数,则输出原数,否则输出变化后的数,并将对应数组指针后移。再对得到的新顺序进行合法判断,若大小关系和要求的完全一致,则输出答案,否则无解。
这题还有更简单的拓扑排序做法,我正在研究,稍后更新。
#pragma comment(linker, "/STACK:1000000000")
#include <bits/stdc++.h>
#define LL long long
#define INF 0x3f3f3f3f
#define IN freopen("E.in","r",stdin);
#define OUT freopen("out.txt", "w", stdout);
using namespace std;
#define MAXN 505
#define MAXM 25005
int s[MAXM], t[MAXM];
int a[MAXN], b[MAXN], pos[MAXN], c[MAXN];
bool dayu[MAXN][MAXN], res[MAXN][MAXN], u[MAXN][MAXN];
bool vis[MAXN];
int main()
{
int T;
scanf("%d", &T);
int cas = ;
while(T--){
cas++;
int n;
scanf("%d", &n);
for(int i = ; i <= n; i++){
scanf("%d", &a[i]);
}
memset(dayu, , sizeof(dayu));
memset(vis, , sizeof(vis));
memset(b, , sizeof(b));
for(int i = ; i <= n; i++){
for(int j = i + ; j <= n; j++){
dayu[a[i]][a[j]] = true;
}
}
int m;
scanf("%d", &m);
memset(u, , sizeof(u));
for(int i = ; i <= m; i++){
scanf("%d%d", &s[i], &t[i]);
if(u[s[i]][t[i]]) continue;
u[s[i]][t[i]] = true;
dayu[s[i]][t[i]] = !dayu[s[i]][t[i]];
dayu[t[i]][s[i]] = !dayu[t[i]][s[i]];
}
int x, y;
int p = ;
bool noans = false;
for(int i = ; i <= m; i++){
if(dayu[s[i]][t[i]]){
x = t[i];
y = s[i];
}else{
x = s[i];
y = t[i];
}
if(!vis[x]){
vis[x] = true;
int j = p;
if(p == ){
b[++p] = x;
}
else{
while(j >= && dayu[x][b[j]]){
j--;
}
for(int k = p; k > j; k--){
pos[b[k]]++;
b[k + ] = b[k];
}
b[j + ] = x;
p++;
}
pos[x] = j + ;
}
if(!vis[y]){
vis[y] = true;
int j = pos[x] - ;
while(j >= && dayu[y][b[j]]){
j--;
}
for(int k = p; k > j; k--){
pos[b[k]]++;
b[k + ] = b[k];
}
p++;
b[j + ] = y;
pos[y] = j + ;
}
}
int j = ;
memset(pos, , sizeof(pos));
for(int i = ; i <= n; i++){
if(vis[a[i]]){
c[i] = b[j++];
}
else{
if(dayu[a[i]][b[j]] || j > p){
c[i] = a[i];
}
else{
c[i] = b[j++];
}
}
pos[c[i]] = i;
}
memset(res, , sizeof(res));
for(int i = ; i <= n; i++){
for(int j = i + ; j <= n; j++){
res[c[i]][c[j]] = true;
}
}
for(int i = ; i <= n; i++){
for(int j = ; j <= n; j++){
if(res[i][j] != dayu[i][j]){
noans = true;
break;
}
}
if(noans){
break;
}
}
if(noans){
printf("IMPOSSIBLE\n");
continue;
}
for(int i = ; i < n; i++){
printf("%d ", c[i]);
}
printf("%d\n", c[n]);
}
return ;
}
最新文章
- 《简明python教程》笔记二
- Android ListView加载更多
- ASP.NET MVC在线人数统计
- iOS 9 failed for URL: ";XXX://@"; - error: ";This app is not allowed to query for scheme XXX"; iOS 从APP里启动另一APP
- 计算机中如何表示数字-07IEEE754浮点数标准
- MYSQL性能查看(命中率,慢查询)
- LinqToExcel常用对象
- 《Windows驱动开发技术详解》之自定义StartIO
- .NET并行处理和并发1-Threads and Theading
- Chrome Extension in CLJS —— 搭建开发环境
- HDU 5783 Divide the Sequence (训练题002 B)
- Design Patterns笔记
- 将id传过去,根据id显示下面的详情页面
- hyperledger fabric 架设命令
- Vue+elementui +Springboot session丢失解决方案
- xshell下mysql数据库只导出表结构不导出数据
- createobjbyreplace(str,arr) js替换方法保存
- WebAPI 操作返回
- qt安装
- LeetCode47.Permutations II(剑指offer38-1)
热门文章
- GenIcam标准(三)
- gps 地图
- root用户无法切换到cdh的hive账号上
- IOS音频架构之Audio Unit
- 支持并发的httpclient(基于tcp连接池以及netty)
- uva_11997,K Smallest Sums优先队列
- Getting Started with MongoDB (MongoDB Shell Edition)
- iOS自动布局高级用法 &;&; 纯代码约束写法
- 智课雅思短语---五、 in contrast / on the contrary
- hdoj--3552--I can do it!(贪心模拟)