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

点击打开链接

题意:给一个无向图,然后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;
}
时间: 2024-08-14 23:21:44

HDU 3663 舞蹈链之不可重复覆盖的相关文章

HDU 3529 舞蹈链之可重复覆盖

点击打开链接 题意:在点的地方放炸弹,炸弹可以炸到的地方无限远并且是十字的形式,但是如果炸到#或者*则这个方向停下来,问最少需要多少炸弹同时引爆才可以将所有#炸掉 思路:一道模版舞蹈链了,对于建图我们将所有的#看做列,然后所有的点也就是放炸弹的地方看做行,然后找一下每个点可以炸的#就可以了,建图后跑个舞蹈链可重复的模版就可以了 #include <vector> #include <stdio.h> #include <string.h> #include <st

HDU 2828 舞蹈链可重复覆盖

点击打开链接 题意:给n个灯,每个灯能保持亮的状态的条件给出,最后问能否将所有的灯都保持亮的状态,能的话输出每个开关的状态 思路:明显的舞蹈链可重复覆盖嘛,列为n个灯,然后行为每个开关开还是关,所以行要有2*m行,注意的是在跑舞蹈链的时候,每一个开关只能有开或关的状态,所以我们要判断一下它对应的状态是否走过 #include <vector> #include <stdio.h> #include <string.h> #include <stdlib.h>

HDU 2295 舞蹈链可重复覆盖+二分

点击打开链接 题意:给了n个点和m个圆心,在用不多于k个圆的情况下,使得所有的点被覆盖,最小的圆半径多大 思路:问最小的什么什么那么用二分准没错,那么二分条件是什么呢,就是用不多于K的圆能否将所有点覆盖,覆盖这部分就是裸的舞蹈链的可重复覆盖,行为m个圆,列为n个点,然后求出的最少行小于等于K,则成立,否则不成立 #include <math.h> #include <vector> #include <stdio.h> #include <string.h>

HUST 1017 舞蹈链之不可重复覆盖

点击打开链接 题意:每一行的某些列给定为1,现在问是否能找某些行,使得每一列的1出现一次 思路:与可重复覆盖的区别就是选定的列的1与其它行的又重复的,那么重复的那行也需要删除,而可重复的不需要删除,那么直接改一改之前的那个模版就可以求这道题了,而且需要输出所选行,任意一个就可以,那么在查找的时候直接赋给一个数组即可 #include <vector> #include <stdio.h> #include <string.h> #include <stdlib.h

FZU 1686 舞蹈链之可重复覆盖

点击打开链接 题意:中文题了 思路:做了几个舞蹈链的题目了,第一感觉总会想到去用网络流去写,哎YY一会就是错的,乖乖用舞蹈链来写把,对于这道题目我们可以将所有的1看成是列,然后行则是一个一个的小矩阵,每一行也就是小矩阵连接自己可以消灭的1,但是首先要给所有的1编一下号,再然后就是模版舞蹈链了 #include <vector> #include <stdio.h> #include <string.h> #include <stdlib.h> #includ

HDU 5046 Airport(DLX重复覆盖)

HDU 5046 Airport 题目链接 题意:给定一些机场,要求选出K个机场,使得其他机场到其他机场的最大值最小 思路:二分+DLX重复覆盖去判断即可 代码: #include <cstdio> #include <cstring> using namespace std; const int MAXNODE = 4005; const int MAXM = 65; const int MAXN = 65; const int INF = 0x3f3f3f3f; int K;

HDU 5046 Airport ( Dancing Links 重复覆盖 )

今年上海网络赛的一道题目 , 跟 HDU 2295 如出一辙 , 就是距离的计算一个是欧几里得距离 , 一个是曼哈顿距离 学完DLX感觉这题好水 ,就是一个裸的重复覆盖 注意下别溢出就行了 #include <stdio.h> #include <string.h> #include <algorithm> #include <vector> #include <math.h> #include <stdlib.h> using na

HDU 3335 Divisibility dancing links 重复覆盖

分析: dlx重复覆盖的巧用,重复覆盖的原理恰好符合本题的筛选方式,即选择一个数后,该数的倍数或约数可以保证在之后的搜索中不会被选择 于是修改一下启发函数,求解最大的重复覆盖即可. 其实不一定不被选择,只是选择以后,要么达不成目标,要不达到目标,也不如不选择更优 举下面的例子 3 2 3 6 答案一看就是 2 初始的dancing links的表示是这样的 2   3   6 2    1   0   1 3    0   1   1 6    1   1   1 然后肯定先选第一列进行删 删

HDU 3335 Divisibility(DLX可重复覆盖)

Problem Description As we know,the fzu AekdyCoin is famous of math,especially in the field of number theory.So,many people call him "the descendant of Chen Jingrun",which brings him a good reputation. AekdyCoin also plays an important role in th