首页 > 其他 > 详细

POJ 3683 Priest John's Busiest Day

时间:2015-10-28 12:17:21      阅读:267      评论:0      收藏:0      [点我收藏+]

2-SAT简单题,判断一下两个开区间是否相交

#include<cstdio>
#include<cstring>
#include<cmath>
#include<queue>
#include<vector>
#include<stack>
#include<algorithm>
using namespace std;

const int maxn=2000+10;
int N,M;
struct Time
{
    int Start;
    int End;
    int cost;
    int S1,E1,S2,E2;
} t[maxn];
char s[1000];

struct TwoSAT
{
    int n;
    vector<int> G[maxn*2];
    bool mark[maxn*2];
    int S[maxn*2],c;

    bool dfs(int x)
    {
        if(mark[x^1]) return false;
        if(mark[x]) return true;
        mark[x]=true;
        S[c++]=x;
        for(int i=0;i<G[x].size();i++)
            if(!dfs(G[x][i])) return false;
        return true;
    }

    void init(int n)
    {
        this->n=n;
        for(int i=0;i<n*2;i++) G[i].clear();
        memset(mark,0,sizeof mark);
    }

    void add_clause(int x,int y)
    {
        G[x].push_back(y^1);
        G[y].push_back(x^1);
    }

    bool solve()
    {
        for(int i=0;i<2*n;i+=2)
            if(!mark[i]&&!mark[i+1])
            {
                c=0;
                if(!dfs(i))
                {
                    while(c>0) mark[S[--c]]=false;
                    if(!dfs(i+1)) return false;
                }
            }
        return true;
    }
};

bool F(int x1,int y1,int x2,int y2)
{
    if(x1<x2&&x2<y1) return 1;//区间相交
    if(x1<y2&&y2<y1) return 1;//区间相交
    if(x2<x1&&x1<y2) return 1;//区间相交
    if(x2<y1&&y1<y2) return 1;//区间相交
    if(x1==x2||y1==y2) return 1;//区间相交
    return 0;//区间不相交
}

int main()
{
    while(~scanf("%d",&N))
    {
        TwoSAT T; T.n=N; T.init(N);
        for(int i=0; i<N; i++)
        {
            int h1,h2,m1,m2,tt;
            
            scanf("%s",s);
            h1=(s[0]-0)*10+(s[1]-0);
            m1=(s[3]-0)*10+(s[4]-0);

            scanf("%s",s);
            h2=(s[0]-0)*10+(s[1]-0);
            m2=(s[3]-0)*10+(s[4]-0);

            scanf("%d",&tt);

            t[i].Start=h1*60+m1;
            t[i].End=h2*60+m2;
            t[i].cost=tt;
            
            t[i].S1=t[i].Start;
            t[i].E1=t[i].Start+t[i].cost;
            
            t[i].S2=t[i].End-t[i].cost;
            t[i].E2=t[i].End;
        }
        
        for(int i=0;i<N;i++)
        {
            for(int j=i+1;j<N;j++)
            {
                if(F(t[i].S1,t[i].E1,t[j].S1,t[j].E1)) 
                    T.add_clause(2*i,2*j);
                if(F(t[i].S1,t[i].E1,t[j].S2,t[j].E2)) 
                    T.add_clause(2*i,2*j+1);    
                if(F(t[i].S2,t[i].E2,t[j].S1,t[j].E1)) 
                    T.add_clause(2*i+1,2*j);
                if(F(t[i].S2,t[i].E2,t[j].S2,t[j].E2)) 
                    T.add_clause(2*i+1,2*j+1);
            }
        }
        if(T.solve()) 
        {
            printf("YES\n");
            for(int i=0;i<N;i++)
            {
                if(T.mark[2*i]) printf("%02d:%02d %02d:%02d\n",t[i].S1/60,t[i].S1%60,t[i].E1/60,t[i].E1%60);
                else  printf("%02d:%02d %02d:%02d\n",t[i].S2/60,t[i].S2%60,t[i].E2/60,t[i].E2%60);
            }
        }
        else printf("NO\n");
    }
    return 0;
}

 

POJ 3683 Priest John's Busiest Day

原文:http://www.cnblogs.com/zufezzt/p/4916749.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!