某考试 T2 Seg

Seg

 

【问题描述】
数轴上有n条线段,第i条线段的左端点是a[i],右端点是b[i]。
Bob发现1~2n共2n个整数点,每个点都是某条线段的端点。
这些线段有如下两类特点:
1 x y,表示第x条线段和第y条线段相交(相交在这里指至少有一个公共点)
2 x y,表示第x条线段在第y条线段的左边,且它们不相交。
共有m个特点,每个特点都是如上两类之一。
Bob想通过这些特点推理得到每条线段的端点。
【输入格式】
第一行两个正整数n,m
接下来m行,每行三个正整数,描述线段的特点,格式见题目描述
【输出格式】
输出n行,第i行两个正整数,用空格隔开,分别是a[i]和b[i]
可能有多种答案,输出字典序最小的答案。即先要求a[1]最小,若仍有多解再要求b[1]最小,若仍有多解再要求a[2]最小,若仍有多解再要求b[2]最小,若仍有多解再要求a[3]最小……
如果无解输出“Wrong”(不输出引号)。
【样例输入】
3 2
1 2 3
2 1 3
【样例输出】
1 2
3 5
4 6
【数据规模和约定】
对于30%的数据, 1<=n,m<=10
对于60%的数据, 1<=n,m<=1000
对于100%的数据,1<=n,m<=100000

 

    题解在注释里有,要注意的就是一点,我们如果要求一个字典序最小的拓扑排序,正确的做法并不是尽量把小的塞到前面,

而是把大的尽量塞到后面。

    这个要理解的话,就是如果我们尽量把小的噻前面,有可能一个优先级很高的是一个优先级很低的后继,这时候因为优先队列是优先出小的

所以那个优先级很高的会很晚出来。

    而我们如果反着跑的话,是不会出现这种情况的,因为优先级低的会被挤到后面去。

 

/*
    可以把每条线段的左右端点都看成一个变量。
	设第i条线段的左右端点的变量为l[i],r[i]。
	
	条件1 => l[x]<r[y] && l[y]<r[x]
	条件2 => r[x]<l[y]
	
	所以我们把大小关系拓扑建图一下,然后跑一个
	优先编号小的线段端点,优先同一线段的左端点的
	拓扑排序,这个用一下优先队列就行了。 
*/
#include<bits/stdc++.h>
#define ll long long
#define pb push_back
#define maxn 200005
using namespace std;
int ans[maxn],cnt=0,now,las;
int n,id[maxn],m,opt,X,Y;
priority_queue<int> q;
vector<int> g[maxn];

inline void topsort(){
	for(int i=1;i<=now;i++) if(!id[i]) q.push(i);
	int x,to;
	while(!q.empty()){
		x=q.top(),q.pop();
		ans[x]=now--;
		for(int i=g[x].size()-1;i>=0;i--){
			to=g[x][i];
			if(!(--id[to])) q.push(to);
		}
	}
}

int main(){
	freopen("seg.in","r",stdin);
	freopen("seg.out","w",stdout);
	scanf("%d%d",&n,&m);
	now=n<<1;
	
	for(int i=1;i<=n;i++){
		g[i*2].pb(i*2-1);
		id[i*2-1]++;
	}
	
	for(int i=1;i<=m;i++){
		scanf("%d%d%d",&opt,&X,&Y);
		if(opt==1){
			g[Y*2].pb(X*2-1),id[X*2-1]++;
			g[X*2].pb(Y*2-1),id[Y*2-1]++;
		}
		else{
			g[Y*2-1].pb(X*2);
			id[X*2]++;
		}
	}
	
	topsort();
	
	if(now) puts("Wrong");
	else for(int i=1;i<=n;i++){
		printf("%d %d\n",ans[i*2-1],ans[i*2]);
	}
	
	return 0;
}

  

posted @ 2018-03-05 11:31  蒟蒻JHY  阅读(246)  评论(0编辑  收藏  举报