Dancing Links 学习笔记
Dancing Links
本周的AI引论作业布置了一道数独
加了奇怪剪枝仍然TLE的Candy?不得不去学了dlx
dlxnb!
Exact cover
设全集X,X的若干子集的集合为S。精确覆盖是指,选择一个S的子集S‘,满足X中的每一个元素在S’中恰好出现一次。
是一个NPC问题。
可以表示成01矩阵形式,选择若干行,使得每一列恰好有且仅有一行为1.
Sudoku
数独可以转化为精确覆盖问题。
令N=81为数独中格子个数,则:
(x, y)=1表示(x,y)处填了数(x+N, z)=1表示x行填了z(y+N*2, z)=1表示y列填了z(r+N*3, z)=1表示r宫填了z
对于已经填了数的格子,转化为1行;
对于空的格子,转化为9行。
Algorithm X
一种显然的dfs:
- 就是选择某一列,再选择该列的为1的某一行。
- 删除该列(包括该列上为1的所有行)
- 删除该行(包括该行上为1的所有列)
一个显然的启发式优化:minimum-remaining-values(MRV) heuristic
- 优先选择节点个数(1的个数)少的列。
Dancing Links
Dancing links is the technique suggested by Donald Knuth to efficiently implement his Algorithm X.
是一种用来高效实现algorithm X的数据结构。
就是“交叉十字循环双向链表”。
第0行分别是root和每一列的列首节点
其他的只有为1的位置才有节点。
删除某一列时,只要处理该列首节点(包括其左右节点)的左右指针;
删除某列时同时要删除该列上为1的所有行;
删除某一行时,只要处理该行所有节点(包括其上下节点)的上下指针。
值得注意的是,删除之后该列/行的结构没有改变。

实现细节
每个节点维护:
l r u d左右上指针col列指针row行标号cnt保存该列的元素个数(只列首/用来MRV优化)
a和h数组保存列首/行首节点指针
初始化init
- 处理列首
在矩阵(r,c)位置加入一个元素/1 link
- 加在
a[c]下,h[r]右 - (实际的“线”是不是直的不重要
删除某列del
- 删除该列,以及该列上的所有行
恢复某列add
- 按删除相反的顺序恢复
主过程dance
root->r == root时完成选择元素最少的某列
c并删除该列(包括该列上为1的所有行)选择该列上为1的某行,删除该行(包括该行上为1的所有列)
实际上这一行在2中已经删除了,只要处理该行的列即可
递归搜索
恢复该行
恢复该列
注意
- del/add时处理个数是必要的,因为那一行所对应的列不一定会被删去
- 恢复要按照删除的逆序
代码
POJ 3076 16*16数独问题的代码
结构体版太丑了还是放指针版吧QwQ
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <ctime>
using namespace std;
const int NUM = 260*4*16, N = 260*16, K = 16, L = 4, M = N*16;
int n = 256*4, num = 256, m=0;
struct meow {
meow *l, *r, *u, *d, *col;
int row;
int cnt;
} pool[NUM];
meow *a[NUM], *h[M], *root;
int ans[N], sz;
char s[20][20];
struct action {
int x, y, z;
} q[M];
void init() {
for(int i=0; i<=n; i++) a[i] = &pool[i];
for(int i=0; i<=n; i++) {
a[i]->l = a[i-1];
a[i]->r = a[i+1];
a[i]->u = a[i]->d = a[i];
a[i]->col = a[i];
a[i]->row = 0;
a[i]->cnt = 0;
}
a[0]->l = a[n]; a[n]->r = a[0];
root = a[0];
sz = n;
memset(h, 0, sizeof(h));
}
void link(int r, int c) {
sz++;
meow *x = a[sz] = &pool[sz];
x->row = r;
x->col = a[c];
a[c]->cnt++;
x->d = a[c]->d; x->d->u = x;
x->u = a[c]; x->u->d = x;
if(h[r] == NULL) {
h[r] = x->l = x->r = x;
}
else {
x->r = h[r]->r; x->r->l = x;
x->l = h[r]; x->l->r = x;
}
}
void del(meow *x) {
x->l->r = x->r;
x->r->l = x->l;
for(meow *i = x->d; i != x; i = i->d)
for(meow *j = i->r; j != i; j = j->r) {
j->d->u = j->u;
j->u->d = j->d;
j->col->cnt--;
}
}
void add(meow *x) {
x->l->r = x->r->l = x;
for(meow *i = x->u; i != x; i = i->u)
for(meow *j = i->l; j != i; j = j->l) {
j->u->d = j->d->u = j;
j->col->cnt++;
}
}
bool dance(int k) {
if(root->r == root) {
for(int i=1; i<=num; i++) {
action &x = q[ans[i]];
s[x.x][x.y] = 'A' + x.z-1;
}
return true;
}
meow *c = root; c->cnt = 1e9;
for(meow *x = root->r; x != root; x = x->r)
if(x->cnt < c->cnt) c = x;
del(c);
for(meow *i = c->d; i != c; i = i->d) {
ans[k+1] = i->row;
for(meow *j = i->r; j != i; j = j->r) del(j->col);
if(dance(k+1)) return true;
for(meow *j = i->l; j != i; j = j->l) add(j->col);
}
add(c);
return false;
}
inline int grid_id(int x, int y, int k=L) {return (x-1)/k*k + (y-1)/k+1;}
void sudoku(int x, int y, int z) {
m++;
link(m, (x-1)*K+y);
link(m, (x-1)*K+z + num);
link(m, (y-1)*K+z + num*2);
link(m, (grid_id(x, y)-1)*K+z + num*3);
q[m] = (action) {x, y, z};
}
int main() {
while(scanf("%s", s[1]+1) != EOF) {
init();
for(int i=1; i<=K; i++) {
for(int j=1; j<=K; j++) {
int a;
if(s[i][j] == '-') a = 0;
else a = s[i][j]-'A'+1;
if(a != 0) sudoku(i, j, a);
else for(int k=1; k<=K; k++) sudoku(i, j, k);
}
if(i != K) scanf("%s", s[i+1]+1);
}
dance(0);
for(int i=1; i<=K; i++) {
for(int j=1; j<=K; j++) printf("%c", s[i][j]);
puts("");
}
puts("");
}
}
Dancing Links 学习笔记的更多相关文章
- POJ 3740 Dancing Links
Dancing Links学习:http://www.cnblogs.com/steady/archive/2011/03/15/1984791.html 以及图文学习:http://www.cnbl ...
- Dancing Links X 学习笔记
\(\\\) Definitions 双向链表:记录前后两个指针的链表,每个顺序关系都有双向的指针维护. \(Dancing\ Links\):双向十字循环链表,建立在二维关系上,每个元素记录上下左右 ...
- [HDU1017]Exact cover[DLX][Dancing Links详解][注释例程学习法]
Dancing Links解决Exact Cover问题. 用到了循环双向十字链表. dfs. 论文一知半解地看了一遍,搜出一篇AC的源码,用注释的方法帮助理解. HIT ACM 感谢源码po主.链接 ...
- ZOJ 3209 Treasure Map (Dancing Links)
Treasure Map Time Limit: 2 Seconds Memory Limit: 32768 KB Your boss once had got many copies of ...
- HUST 1017 - Exact cover (Dancing Links 模板题)
1017 - Exact cover 时间限制:15秒 内存限制:128兆 自定评测 5584 次提交 2975 次通过 题目描述 There is an N*M matrix with only 0 ...
- javascripts学习笔记(五):用js来实现缩略语列表、文献来源链接和快捷键列表。
1 缩略语列表问题出发点:一段包含大量缩略语的文本,例如: <p> The <abbr title="World Wide Web Consortium"> ...
- Makefile的学习笔记
Makefile的学习笔记 标签: makefilewildcard扩展includeshellfile 2012-01-03 00:07 9586人阅读 评论(2) 收藏 举报 分类: Linux ...
- X-Cart 学习笔记(一)了解和安装X-Cart
目录 X-Cart 学习笔记(一)了解和安装X-Cart X-Cart 学习笔记(二)X-Cart框架1 X-Cart 学习笔记(三)X-Cart框架2 X-Cart 学习笔记(四)常见操作 一.了解 ...
- <老友记>学习笔记
这是六个人的故事,从不服输而又有强烈控制欲的monica,未经世事的千金大小姐rachel,正直又专情的ross,幽默风趣的chandle,古怪迷人的phoebe,花心天真的joey——六个好友之间的 ...
随机推荐
- 超简单CentOS7 配置阿里云yum源
1.打开centos的yum文件夹 输入命令cd /etc/yum.repos.d/ 2.用wget下载repo文件 输入命令wget http://mirrors.aliyun.com/repo ...
- Java装箱的 " == " 的问题
装箱和拆箱 packagecom.xzj.Test; /* * @ author thisxzj * @ create 2019-02-25 10:56 */ publicclassBase{ ...
- mysql并发控制之数据库锁
1.mysql和redis的区别 mysql是一种关系型数据库,数据会最终存储在磁盘上.而redis是一种非关系型的nosql数据库,以key-value的形式存储数据,将数据存储在内存.从性能上来说 ...
- JavaScript的数组和循环
1. 数组: a) 声明数组:var 数组名 = new Array(数组大小); Var emp=new Array(“AA”,“BB”,“CC”): b) 添加 ...
- Java 实现TCP/IP协议的收发数据(服务端)
功能如下: 注: 只有服务端,没有客户端,测试时采用第三方软件作为客户端的. 收发数据目前能正常收发数据,只是中文的会变成乱码显示. 采用Thread类实现一个收发数据的线程. 服务端代码: impo ...
- 20175333曹雅坤 实验二 Java面向对象程序设计
实验二 Java面向对象程序设计 实验内容 1. 初步掌握单元测试和TDD 2. 理解并掌握面向对象三要素:封装.继承.多态 3. 初步掌握UML建模 4. 熟悉S.O.L.I.D原则 5. 了解设计 ...
- 【C++11】unoedered_map和map(部分转载)
1.结论 新版的hash_map都是unordered_map了,这里只说unordered_map和map. 运行效率:unordered_map最高,而map效率较低但提供了稳定效率和有序的序列. ...
- qml layout
一.使用总结 1.锚点:锚点锚在父控件左边anchors.left: parent.left ,才可以设置anchors.leftMargin:20 才有作用,设置anchors.topMargin: ...
- Gitlab_ansible_jenkins三剑客③Ansible的安装及使用
一台服务器可能会安装不同的python应用,不同的应用可能使用的模块版本不同,如果都安装在同样的环境下容易冲突,为了避免冲突,引入virtualenv 这个包管理工具进行环境的隔离 使用pip安装之前 ...
- SSM增删改查
闲着无聊配置一遍SSM以及添加功能增删改查,如下图,其中坎坷也挺多!!! 1.工程如下图.(请忽略红叉,没有错误) 2.首先配置pom.xml文件. <project xmlns="h ...