zoukankan      html  css  js  c++  java
  • 【CF827F】Dirty Arkady's Kitchen

    题目

    题目链接:https://codeforces.com/contest/827/problem/F
    给定一张无向图,每条边在时间 ([l_i,r_i)) 才能通过,通过花费 (1) 的时间。你不能在原地停留。
    你在时刻 (0)(1) 号点,求最快何时能到达 (n) 号点。
    无法到达或被迫停留,输出 -1
    (n,mleq 5 imes 10^5)

    思路

    因为可以在一条边上反复横跳,所以如果把每一个点拆成奇点和偶点的话,能到达的时间可以看作是若干个区间的交。
    因为每一个点都拆成了两个点,那么原来的一条无向边就能拆成 (4) 条有向边。把有向边按照 (l) 排序。
    (f[x]) 表示可以到达点 (x) 的最晚时间。那么对于一条边 ((u,v,l,r)),可以在第 (l+1) 时刻到达 (v) 当且仅当现在 (f[u]geq l) 的。如果不能直接到达,我们就把这一条边扔进 (u) 的一个 vector 内。
    如果可以在 (l+1) 时刻到达,我们就把 (f[v]) 更新为 (max(f[v],r+1)),然后把在 (v) vector 内的边都处理掉。因为此时这些边一定都可以走。注意这些边的 (l) 都需要更改为 (l+1)
    时间复杂度 (O(nlog n))

    代码

    #include <bits/stdc++.h>
    using namespace std;
    
    const int N=1000010;
    int n,m,f[N];
    
    struct node
    {
    	int u,v,l,r;
    	
    	friend bool operator <(node x,node y)
    	{
    		return x.l>y.l;
    	}
    };
    vector<node> e[N];
    priority_queue<node> q;
    
    int main()
    {
    	scanf("%d%d",&n,&m);
    	if (n==1) return printf("0"),0;
    	for (int i=1,u,v,l,r;i<=m;i++)
    	{
    		scanf("%d%d%d%d",&u,&v,&l,&r);
    		r--;
    		q.push((node){u,v+n,l+(l&1),r-(r&1)});
    		q.push((node){v,u+n,l+(l&1),r-(r&1)});
    		q.push((node){u+n,v,l+!(l&1),r-!(r&1)});
    		q.push((node){v+n,u,l+!(l&1),r-!(r&1)});
    	}
    	memset(f,-1,sizeof(f));
    	f[1]=0;
    	while (q.size())
    	{
    		node u=q.top(); q.pop();
    		if (u.l>u.r) continue;
    		if (u.l>f[u.u]) { e[u.u].push_back(u); continue; }
    		if (u.v==n || u.v==2*n) return printf("%d",u.l+1),0;
    		f[u.v]=max(f[u.v],u.r+1);
    		for (int i=0;i<(int)e[u.v].size();i++)
    		{
    			node v=e[u.v][i];
    			v.l=u.l+1; q.push(v);
    		}
    		e[u.v].clear();
    	}
    	cout<<"-1";
    	return 0;
    }
    
  • 相关阅读:
    快速模幂
    UPC-2249 曲线分割【递推】
    maven 服务器
    maven repo
    php-fpm sock
    mysql
    go 1
    xdebug
    centos
    win10 2503 2502
  • 原文地址:https://www.cnblogs.com/stoorz/p/15399115.html
Copyright © 2011-2022 走看看