gym101201F Illumination 2-SAT

题目传送门

题目大意:

  给出n*n的网格,l栈灯,每盏灯可以选择照亮竖着的2*r+1的范围,或者横着的2*r+1的范围,要求一个格子不会同时被一盏以上的横着的灯照亮,也不能被一盏以上的竖着的灯照亮,所有灯必须亮着,问是否可行。

  思路:2-sat的题目,如果两盏灯之间不能同时横着放,则代表必须一横一竖,则找到了2-sat的建边关系,具体看代码。

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn = 1010;
int n,r,l,head[maxn<<1],tot;
bool vis[maxn<<1];
int s[maxn<<1],top;
struct node{
    int x,y;
}a[maxn];
struct edge{
    int to,Next;
}e[(maxn*maxn)<<1];
void init(){
    memset(head,-1,sizeof(head));
    memset(vis,0,sizeof(vis));
    tot=0;
}
void addv(int u,int v,int val){
    int tmpu=(u<<1)+val;
    int tmpv=(v<<1)+val;
    e[++tot]={tmpv,head[tmpu^1]},head[tmpu^1]=tot;//注意建边 相容的情况建边 
    e[++tot]={tmpu,head[tmpv^1]},head[tmpv^1]=tot;
}
bool dfs(int u){
    if(vis[u^1])return false;
    if(vis[u])return true;
    vis[u]=1;
    s[top++]=u;
    for(int i=head[u];i!=-1;i=e[i].Next)
    {
        if(!dfs(e[i].to))return false;
    }
    return true;
}
bool two_sat(){
    for(int i=0;i<(l<<1);i+=2){
        if(!vis[i]&&!vis[i^1]){
            top=0;
            if(!dfs(i)){
                while(top)vis[s[--top]]=0;
                if(!dfs(i^1))return false;
            }
            
        }
    }
    return true;
}
int main(){
    while(scanf("%d%d%d",&n,&r,&l)!=EOF)
    {
        init();
        for(int i=0;i<l;i++)
        {
            scanf("%d%d",&a[i].x,&a[i].y);
        }
        for(int i=0;i<l;i++)
        {
            for(int j=0;j<i;j++)
            {
                if(a[i].x==a[j].x&&abs(a[j].y-a[i].y)<2*r+1){
                    addv(i,j,0);
                };
                if(a[i].y==a[j].y&&abs(a[j].x-a[i].x)<2*r+1){
                    addv(i,j,1);
                };
            }
        }
        if(two_sat())puts("YES");
        else puts("NO");
    }
}
View Code
原文地址:https://www.cnblogs.com/mountaink/p/10093881.html