【Luogu_P1983】车站分级

链接

思路:

把没有停靠的车站向停靠了的车站连一条边,然后在拓扑排序


#include
#include
#include
#include
#include
#include
#include
using namespace std;

int n, m, tot;
int pt[10010], now[1010], d[1010], f[100100];
int v[10010], ans[10010], head[10010], ma[1001][1001];
struct node
{
	int to, next;
}a[100010];

void VL_ljb(int x, int y)
{
    a[++tot]=(node){y, head[x]};
    head[x]=tot;
}

void VL_toupu()
{
	int hd=0, tl=0;
    for(int i=1; i<=n; i++)
    {
    	if(d[i]==0)
    	{
    		tl++;
    		f[tl]=i;
    		ans[i]=1; 
    	}
    }
    while(hd<tl)
    {
    	hd++;
    	int x=f[hd];
    	for(int i=head[x]; i; i=a[i].next)
    	{
    		int y=a[i].to;
    		d[y]--;
    		if(d[y]<=0)
    		{
    			ans[y]=ans[x]+1;
    			f[++tl]=y;
    		}
    	}
    }
}
int main(){
	scanf("%d%d", &n, &m);
	for(int i=1; i<=m; i++)
	{
		memset(now, 0, sizeof(now));
		int s;
		scanf("%d", &s);
		for(int j=1; j<=s; j++)
		{
			scanf("%d", &pt[j]);
			now[pt[j]]=1;
		}
		for(int j=pt[1]; j<=pt[s]; j++)
		{
			if(now[j]==1) continue;
			for(int k=1; k<=s; k++)
				ma[j][pt[k]]=1;
		}
	}
	for(int i=1; i<=n; i++)
		for(int j=1; j<=n; j++)
			if(ma[i][j]==1)
				VL_ljb(i, j), d[j]++;
	VL_toupu();
	int ans1=0;
	for(int i=1; i<=n; i++)
		ans1=max(ans1, ans[i]);
    printf("%d", ans1);
	return 0;    
}

你可能感兴趣的:(题解,拓扑排序)