洛谷 P4929 【模板】舞蹈链(DLX)

题目链接

DLX例题(洛谷模板题就是省选难度了qwq)。

DLX的具体思想我就不赘述了,网上有很多讲的不错的。下附链接

网站1网站2网站3网站4

AC代码如下(代码思路就看注释吧……)

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
#define mx 250505 
using namespace std;
int n,m,cnt;
int l[mx],r[mx],u[mx],d[mx],row[mx],col[mx];//定义一个点的左右上下以及行数列数。
int h[mx],s[mx];//一行的第一个数 ,一列有几个数
int ansk[mx];//答案 
inline void Init(int m){//初始化m列。m+1个数,0为head(无实际意义,是最后用来判断的 
    for(int i=0;i<=m;i++){
        r[i]=i+1;
        l[i]=i-1;
        u[i]=d[i]=i; 
    }
    r[m]=0,l[0]=m;
    memset(h,-1,sizeof(h));
    memset(s,0,sizeof(s));
    cnt=m+1;//之后录入的数从m+1的下标开始 
}
inline void link(int R,int C){//插入一个在R行C列的点 
    s[C]++;
    row[cnt]=R,col[cnt]=C;//记录行和列 
    u[cnt]=C;//cnt的上面为c,即1~m 
    d[cnt]=d[C];//cnt的下面是原先C的下面 
    u[d[C]]=cnt;//原先c的下面的上面是cnt 
    d[C]=cnt; //c的下面是cnt。一系列操作即将cnt插入C和C下面一个数之间,让cnt成为最接近c的那个数(这样写的话最终其实u是在下面的意思的感觉……
    if(h[R]==-1)//如果这一行没有数的话就将该数作为第一个数录入 
        h[R]=r[cnt]=l[cnt]=cnt;
    else{//有数的话 
        r[cnt]=h[R];//将cnt的右边连接到h[R]的左边 
        l[cnt]=l[h[R]];//将cnt的左边连接到h[R]的左边 
        r[l[h[R]]]=cnt;//将原来h[R]的左边连接到cnt的左边 
        l[h[R]]=cnt;//将cnt连接到h[R]的左边(感觉这样写也是左右颠倒的感觉……不过倒是不影响 
    }     
     cnt++;
     return;
}
inline void remove(int C){//删除c列 
    r[l[C]]=r[C],l[r[C]]=l[C];//删除这一列的存在。
    for(int i=d[C];i!=C;i=d[i])//找每个在该列上的数 
        for(int j=r[i];j!=i;j=r[j]){//找该列上的数所在的那行 ,并删除整行 
            u[d[j]]=u[j];
            d[u[j]]=d[j];
            s[col[j]]--;//该数所在列内数量-1
        } 
    return;
} 
inline void resume(int C){
    r[l[C]]=C,l[r[C]]=C;
    for(int i=d[C];i!=C;i=d[i])
        for(int j=r[i];j!=i;j=r[j]){
            u[d[j]]=j;
            d[u[j]]=j;
            s[col[j]]++; 
        }
    return;
} 
bool dance(int deep){
    if(r[0]==0){//如果列删光了且除了head没有元素留下来。则表示该情况成立 
        for(int i=0;i<deep;i++)
            printf("%d ",ansk[i]);
            return 1;
    }
    int c=r[0];
    for(int i=r[0];i!=0;i=r[i])//优先从列中数最少的开始找。缩小查找范围 
        if(s[i]<s[c])
            c=i;
    remove(c);//删除c列 
    for(int i=d[c];i!=c;i=d[i]){//从c列当中选1行进行查找 
        ansk[deep]=row[i];//记录当前被删的行数
        for(int j=r[i];j!=i;j=r[j]) //加上这句话后感觉remove函数中的第三个循环没什么必要的样子…… 
            remove(col[j]);
        if(dance(deep+1))
            return 1;
        for(int j=l[i];j!=i;j=l[j])
            resume(col[j]);
    } 
    resume(c);
    return 0;
}
int main(){
    scanf("%d%d",&n,&m);
    Init(m); 
    char temp;
    int k;
    for(int i=1;i<=n;i++)
        for(int j=1;j<=m;j++){
            scanf("%d",&k);
             if(k)
                 link(i,j);
        }
    if(!dance(0))
        printf("No Solution!");
    return 0;
}

 

上一篇:P2455 [SDOI2006]线性方程组


下一篇:[LeetCode] 1026. Maximum Difference Between Node and Ancestor 结点与其祖先之间的最大差值