[agc017e]Jigsaw

邓浩漫
2023-12-01

题目大意

给你n块积木,每块积木由三列构成,每块中间那列最长,为固定的H;对于每一块i,左边一列底部会比中间底部高c[i],然后长度为a[i],右边类似地,d[i],b[i].
现在要求你把积木拼接起来,使得所有积木中列底部在同一水平线上,左右两列要么在这条水平线上,要么紧贴着另一块积木某一侧的顶端。
判断是否能够这样拼。
n≤1e5,H≤200,a,b>0,a+c,b+d≤H.

解题思路

我们发现,当一块积木c>0的时候,a的取值没有用。类似地有d,b,那么我们可以把积木分个类。
对于一块积木,若c=0,令l=a,否则l=-c;若d=0,令r=-b,否则r=d;我们称这块积木的类型为(l,r)。
图论转化:连有向边l->r,那么一个合法的拼接是一条路径,而且起点s>0,终点t<0。
那么我们现在要判断,对于一幅包含-H…-1和1…H的图,是否能够拆分成若干段这样的路径。
那么要满足条件:
1,对于i>0,in[i]<=out[i];
2,对于i<0,in[i]>=out[i];
3,对于一个弱联通分量,必须有一个点满足in[i]≠out[i];
前两个条件很显然,后一个条件保证了你在构造真实解的时候,可以把不小心弄出来的环和其他的路径合并,最终构造出合法解。

代码

#include<cstdio> 
#include<algorithm>
#include<cstring>
#include<cmath>
using namespace std;
#define fo(i,j,k) for(i=j;i<=k;i++)
#define fd(i,j,k) for(i=j;i>=k;i--)
#define cmax(a,b) (a=(a>b)?a:b)
#define cmin(a,b) (a=(a<b)?a:b)
typedef long long ll;
const int N=1e6+5,M=2e6+5,mo=998244353;
int fa[N],n,H,A,B,C,D,i,pp,pd[N],chu[N],ru[N],l,r,act[N];
int get(int x)
{
    if (fa[x]==x) return x;
    return fa[x]=get(fa[x]);
}
int main()
{
    scanf("%d %d\n",&n,&H);
    fo(i,0,H*2) fa[i]=i;
    fo(i,1,n)
    {
        scanf("%d %d %d %d",&A,&B,&C,&D);
        if (!C) l=A;else l=-C; 
        if (!D) r=-B;else r=D;
        l+=H;
        r+=H;
        //e=(l,r)
        act[l]=act[r]=1;
        chu[l]++;
        ru[r]++;
        fa[get(l)]=get(r);
    }
    pp=1;
    // neg: in>=out
    fo(i,0,H) if (ru[i]<chu[i]) pp=0;
    // pos: in<=out
    fo(i,H+1,2*H) if (ru[i]>chu[i]) pp=0;
    fo(i,0,2*H) pd[get(i)]|=(ru[i]!=chu[i]);
    pd[H]=1;
    fo(i,0,2*H) if (act[i]&&!pd[get(i)]) pp=0;
    if (pp) printf("YES");else
    printf("NO");
}
 类似资料: