http://www.cnblogs.com/grenet/p/3145800.html

链接给的博客写的很好,比较好懂。

可惜不是c语言...

于是决定自己要建一个模板。

一道裸题:hustoj 1017 exact cover 输入一个矩阵求选哪些行,使得这是一个精确覆盖

AC通道:http://acm.hust.edu.cn/problem/show/1017

 #include<cstdio>
#include<cstring>
#include<algorithm> using namespace std; const int maxn=;
const int INF=0x7fffffff; typedef int array[maxn];
typedef int Array[maxn*maxn]; int n,m,cnt,size;
array last,f,d,ans;
Array L,R,U,D,c,r,s; void build(){ //初始化
/*构造矩阵第一行:
1.将第一行的两端环状处理。[提示]:第一行的元素的标号就是其列的标号。
2.对于第一行的所有元素:
连接其左右;标注其行列;清空列中表示的元素个数;清空表示列的上一个元素的d数组。*/ L[]=m,R[m]=;
for(int i=;i<=m;i++){
L[i]=i-,R[i-]=i;
c[i]=i;r[i]=;
s[i]=;
d[i]=;
} //清空所有行上的表示上一个元素的last数组和表示首元素的f数组
for(int i=;i<=n;i++) last[i]=f[i]=;
} void del(int col){ //删列操作
/*
1.在第一行中删除这个列节点。
2.找到每个在这条列上的节点i
将节点i所在的行从列表中删除
*/ L[R[col]]=L[col],R[L[col]]=R[col]; //在第一行中删除这个列节点。 for(int i=D[col];i!=col;i=D[i])
for(int j=R[i];j!=i;j=R[j]) //将节点i所在的行删除干净[因为是环状的,所以可以删掉这行上的所有节点]
U[D[j]]=U[j],D[U[j]]=D[j],s[c[j]]--; //这些节点都在列表中不再被查询得到。
} void add(int col){ //恢复列操作
/*
1.在第一行中恢复这个列节点。
2.找到每个在这条列上的节点i
将节点i所在的行有顺序的从列表中恢复
*/ R[L[col]]=col,L[R[col]]=col; //因为删除时其实是间接删除[将其两边元素的指针改变],恢复只需要从它自己出发恢复两边即可。 for(int i=U[col];i!=col;i=U[i])
for(int j=L[i];j!=i;j=L[j]) //同删除的顺序相反的添加回来[即后删去的先添加],才保证了顺序,不会错
U[D[j]]=j,D[U[j]]=j,s[c[j]]++; //同删除的感觉,通过行表找到这些丢失的点,然后从它自己来恢复列表中它们的位置。
} bool search(int k){
if(R[]==){ //第一行中的所有元素都被删除[即列都被标记,完成了精确覆盖]了
printf("%d",k);
for(int i=;i<=k;i++)
printf(" %d",r[ans[i]]);
putchar('\n');
return true;
}
int Min=INF,C;
for(int i=R[];i;i=R[i])
if(Min>s[i]) Min=s[i],C=i; //优先选择列中元素少的列来处理
del(C);
for(int i=D[C];i!=C;i=D[i]){ //确定要删掉这列,但是要考虑删掉哪一行[当然是必须要与这一列相交的行]
ans[k+]=i;
for(int j=R[i];j!=i/*环状链表的好处,可以循环寻找,找到所有元素*/;j=R[j]) del(c[j]); //当确定删掉i行时,还要删掉所有与其相交的列
if(search(k+)) return true;
for(int j=L[i];j!=i;j=L[j]) /*环状链表的另一个好处,可以让添加的顺序与删除的顺序相反,从而达到后删去的先添加的形式*/
add(c[j]); //回溯过程,补充回原来删去的列。
}
add(C);
return false;
} void link(int row/*行*/,int col/*列*/){
size++; //点的数目++
s[col]++; //所在列的元素个数++ /*行操作:
如果这一行之前没有元素了,这个元素作为本行的首元素,记录在f[row]中;并记录在表示上一个元素的last[row]中。
若所在行之前还有元素,将这个元素与上一个元素连起来,更新last[];并将这个元素的右端与该行的首元素连起来,形成一个环状的结构。*/
if(!last[row])
last[row]=size,
R[size]=size,L[size]=size,
f[row]=size;
else
L[size]=last[row],R[last[row]]=size,
last[row]=size,
R[size]=f[row],L[f[row]]=size; /*列操作:
【与行操作有所不同】:我们事先已经构造出了矩阵的第一行,不存在所谓首元素,因为首元素就是这一列的标号。
如果这一列之前没有添加过元素,将当前元素与这一列的标号首尾相连,更新表示上一个元素的d[col]。
若之前添加过元素,将这个元素与上一个元素连起来;并将这个元素充当此行的末尾,与这一列的标号相连构成一个环状结构;更新d[col]。*/
if(!d[col])
U[col]=size,D[size]=col,
D[col]=size,U[size]=col,
d[col]=size; else
U[col]=size,D[size]=col,
U[size]=d[col],D[d[col]]=size,
d[col]=size; c[size]=col,r[size]=row; //处理完与其它节点的关系后,再给自己打上行列的标号
} int main(){
int k,x; while(~scanf("%d%d",&n,&m)){
build();
size=m; //前m个元素已经使用过了,不参与计数
for(int i=;i<=n;i++){
scanf("%d",&k);
while(k--)
scanf("%d",&x),link(i,x);
}
if(!search()) puts("NO");
} return ;
}

dancing link的更多相关文章

  1. Dancing Link 详解(转载)

    Dancing Link详解: http://www.cnblogs.com/grenet/p/3145800.html Dancing Link求解数独: http://www.cnblogs.co ...

  2. dancing link模板

    #include<cstdio> #include<iostream> #include<cstring> #include<algorithm> #i ...

  3. dancing link 学习资源导航+心得

    dancing link简直是求解数独的神器,NOIP2009最后一题靶形数独,DFS 各种改变搜索顺序 都没法过,最后还是用了卡时过得.用dancing link写,秒杀所有数据,总时间才400ms ...

  4. 【Dancing Link专题】解题报告

    DLX用于优化精确覆盖问题,由于普通的DFS暴力搜索会超时,DLX是一个很强有力的优化手段,其实DLX的原理很简单,就是利用十字链表的快速删除和恢复特点,在DFS时删除一些行和列以减小查找规模,使得搜 ...

  5. bzoj 2706: [SDOI2012]棋盘覆盖 Dancing Link

    2706: [SDOI2012]棋盘覆盖 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 255  Solved: 77[Submit][Status] ...

  6. hustoj 1017 - Exact cover dancing link

    1017 - Exact cover Time Limit: 15s Memory Limit: 128MB Special Judge Submissions: 5851 Solved: 3092 ...

  7. hdu 1426 Sudoku Killer ( Dancing Link 精确覆盖 )

    利用 Dancing Link 来解数独 详细的能够看    lrj 的训练指南 和 < Dancing Links 在搜索中的应用 >这篇论文 Dancing Link 来求解数独 , ...

  8. Dancing Link专题

    一些链接: http://www.cnblogs.com/-sunshine/p/3358922.html http://www.cnblogs.com/grenet/p/3145800.html 1 ...

  9. Dancing Link --- 模板题 HUST 1017 - Exact cover

    1017 - Exact cover Problem's Link:   http://acm.hust.edu.cn/problem/show/1017 Mean: 给定一个由0-1组成的矩阵,是否 ...

随机推荐

  1. 使用maven, myeclipse工具构建spring mvc项目

    一.使用myeclipse 创建一个新的 maven项目. (ps:1.在filter过滤的时候输入 webapp 选择"maven-archetype-webapp". 2.在m ...

  2. Nginx配置:http重定向,URLRewrite,一个简单框架的配置思路

    一个重定向的应用配置: server { listen       8000; server_name  localhost; root F:/home/projects/test; index   ...

  3. mysql中存不进去json_encode格式的数据

    主要是因为json_encode格式的数据,中间带有\,在存入数据库的时候,会把反斜杠删除了. 所以,想要存进去的话,需要在外层调用一下函数addslashes();这个函数会在每个反斜杠的前面添加反 ...

  4. about the pageload and page init event

    Page_Init The Page_Init event is the first to occur when an ASP.NET page is executed. This is where ...

  5. Oracle 直方图理论

    一.何为直方图 直方图是一种几何形图表,它是根据从生产过程中收集来的质量数据分布情况,画成以组距为底边.以频数为高度的一系列连接起来的直方型矩形图,如图所示 二.ORACLE 直方图 在Oracle中 ...

  6. STM32F0xx_GPIO配置详细过程

    前言 对于初学STM32的人来说,很多基础的知识没有掌握,这些基础知识就成为阻挡他们入门的门槛.因此,今天也把基础的知识分享出来,带领那些还没有迈过这个门槛的人入门. 今天总结“GPIO配置详细”,以 ...

  7. angularjs2 学习笔记(二) 组件

    angular2 组件 首先了解angular2 组件的含义 angular2的应用就是一系列组件的集合 我们需要创建可复用的组件供多个组件重复使用 组件是嵌套的,实际应用中组件是相互嵌套使用的 组件 ...

  8. .net 内存分配及垃圾回收总结

        生存期垃圾回收器 目前有很多种类型的垃圾回收器.微软实现了一种生存期垃圾回收器(Generation Garbage Collector). 生存期垃圾回收器将内存分为很多托管堆,每一个托管堆 ...

  9. 删除字符串第一个byte

    删除字符串第一个byte   一种方式:   char * mag; char buff[1000]; char number; memcpy((char *)msg,buff,len); strnc ...

  10. SQLServer异常捕获

    在SQLserver数据库中,如果有很多存储过程的时候,我们会使用动态SQL进行存储过程调用存储过程,这时候,很可能在某个环节就出错了,但是出错了我们很难去跟踪到出错的存储过程,此时我们就可以使用异常 ...