码迷,mamicode.com
首页 > 其他好文 > 详细

HDU 3663 舞蹈链之不可重复覆盖

时间:2016-06-21 07:08:47      阅读:215      评论:0      收藏:0      [点我收藏+]

标签:

点击打开链接

题意:给一个无向图,然后n个城市的供电范围,每个现在要求每一个城市的D天都可以有电,对于城市A发电,那么与它相邻的所有城市都会有电,但是问题是每个城市一天内只可以被供电一次,否则会坏掉,并且每个城市的供电天数有范围而且每个城市只能开启开关一次,之后不能在使用,也就是说城市A的供电时间必须是连续的,还有就是可以不用这个城市

思路:因为每个城市一天只能被供电一次,那么就是不可重复覆盖的舞蹈链模型了,那么考虑的就是把什么作为列和行了,很明显这道题就是N*D+N作为列,N*D是n个城市的D天都要供电,然后后面的N是保证每个城市直发电一次,然后考虑什么作为行,对于一个城市的发电范围,我们可以截取其中所有的小区间来作为行,当然不用的情况要再加一个0 0,比如1到5的范围便可以变成16行,0 0,1 1,1 2,1 3,1 4,1 5,2 2,2 3,2 4,2 5,3 3,3 4,3 5,4 4,4 5,5 5,对于每一个范围它们连的列就是自己对应的天数的城市,就是一些小细节的处理了,大致就是这样了,对了千万不要忘记重边,重边,重边,主要的事说三遍

#include <vector>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <iostream>
#include <algorithm>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
const int inf=0x3f3f3f3f;
const ll INF=0x3f3f3f3f3f3f3f3fll;
const int maxn=1010;
int L[maxn*400],R[maxn*400],U[maxn*400],D[maxn*400];//节点的上下左右四个方向的链表
int C[maxn*400],H[maxn],cnt[maxn],ans[maxn],Row[maxn*400];//C列H行cnt列链表中元素个数
int n,m,id,len,d;
vector<int>G[maxn];
void init(int lll){
    for(int i=0;i<maxn;i++) G[i].clear();
    for(int i=0;i<=lll;i++){
        cnt[i]=0;U[i]=D[i]=i;
        L[i+1]=i;R[i]=i+1;
    }
    R[lll]=0;id=lll+1;
    memset(H,-1,sizeof(H));
}
void Link(int r,int c){
    cnt[c]++;C[id]=c;Row[id]=r;
    U[id]=U[c];D[U[c]]=id;
    D[id]=c;U[c]=id;
    if(H[r]==-1) H[r]=L[id]=R[id]=id;
    else{
        L[id]=L[H[r]];R[L[H[r]]]=id;
        R[id]=H[r];L[H[r]]=id;
    }
    id++;
}
void Remove(int Size){
    L[R[Size]]=L[Size];
    R[L[Size]]=R[Size];
    for(int i=D[Size];i!=Size;i=D[i]){
        for(int j=R[i];j!=i;j=R[j]){
            U[D[j]]=U[j];D[U[j]]=D[j];
            cnt[C[j]]--;
        }
    }
}
void Resume(int Size){
    for(int i=D[Size];i!=Size;i=D[i]){
        for(int j=R[i];j!=i;j=R[j]){
            U[D[j]]=j;D[U[j]]=j;
            cnt[C[j]]++;
        }
    }
    L[R[Size]]=Size;R[L[Size]]=Size;
}
int Dance(int k){
     int c;
     if(!R[0]){
         len=k;
         return 1;
     }
     for(int Min=inf,i=R[0];i;i=R[i])
         if(Min>cnt[i]) Min=cnt[i],c=i;
     Remove(c);
     for(int i=D[c];i!=c;i=D[i]){
         ans[k]=Row[i];
         for(int j=R[i];j!=i;j=R[j]) Remove(C[j]);
         if(Dance(k+1)) return 1;
         for(int j=L[i];j!=i;j=L[j]) Resume(C[j]);
    }
    Resume(c);
    return 0;
}
int A[70],B[70],tmp[maxn][2];
void Linkline(int l,int r,int x,int kkk){
    Link(kkk,n*d+x);
    for(int i=l;i<=r;i++){
        Link(kkk,(i-1)*n+x);
        for(unsigned j=0;j<G[x].size();j++){
            int t=G[x][j];
            Link(kkk,(i-1)*n+t);
        }
    }
}
int visedge[70][70];
int main(){
    int u,v;
    while(scanf("%d%d%d",&n,&m,&d)!=-1){
        init(n*d+n);
        memset(visedge,0,sizeof(visedge));
        for(int i=1;i<=m;i++){
            scanf("%d%d",&u,&v);
            if(visedge[u][v]) continue;
            G[u].push_back(v);
            G[v].push_back(u);
            visedge[u][v]=1;visedge[v][u]=1;
        }
        for(int i=1;i<=n;i++) scanf("%d%d",&A[i],&B[i]);
        int kkk=1;
        for(int i=1;i<=n;i++){
            tmp[kkk][0]=0;tmp[kkk++][1]=0;
            Link(kkk-1,n*d+i);
            for(int j=A[i];j<=B[i];j++){
                for(int l=j;l<=B[i];l++){
                    tmp[kkk][0]=j;tmp[kkk++][1]=l;
                    Linkline(j,l,i,kkk-1);
                }
            }
        }
        int flag=Dance(0);
        if(flag==0) printf("No solution\n");
        else{
            sort(ans,ans+len);
            for(int i=0;i<len;i++) printf("%d %d\n",tmp[ans[i]][0],tmp[ans[i]][1]);
        }
        printf("\n");
    }
    return 0;
}

HDU 3663 舞蹈链之不可重复覆盖

标签:

原文地址:http://blog.csdn.net/dan__ge/article/details/51713563

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!