struct dl{
// x: line, y: column
struct node{
int c, left, right, up, down;
};
vector<node> a; using vec = vector<int>;
using mat = vector<vec>; vec cnt; // 构造十字交叉循环链表有多种写法
// 1.流行的模板: link(int r, int c),每次添加一个元素。另外需要个数组row[],存储每一行中某个元素(可以是这一行中的任意元素,不必是最后一次添加的那个元素)的位置。
// 2.每次传入0-1矩阵中某一行的列编号数组
// 3.每次传入0-1矩阵的所有元素(我的实现即如此) // 行列都从1开始编号,若不需要求答案则node中不用记录行编号
// m:0-1矩阵的列数
void build(const mat &b, int m){
a.clear();
cnt = vec(m+1);
for(int i=0; i<=m; i++)
a.push_back({i, (i+m)%(m+1), (i+1)%(m+1), i, i}); for(auto &i: b)
for(auto &j: i){
node cur={j, -1, -1, a[j].up, j};
a[cur.up].down=a.size();
a[cur.down].up=a.size(); if(j==i.front())
cur.left=cur.right=a.size();
else{
cur.left=a.size()-1;
cur.right=a.back().right;
a[cur.right].left=a.size();
a[cur.left].right=a.size();
}
a.push_back(cur);
++cnt[j];
}
} // to cover some column
// no overhead!
void cover(int col){
// 删除第 col 列
a[a[col].right].left=a[col].left;
a[a[col].left].right=a[col].right;
// cnt[col]=0 // no need to do so.
// 删除所有在第 col 列有元素的行
for(int i=a[col].down; i!=col; i=a[i].down){
for(int j=a[i].right; j!=i; j=a[j].right){
a[a[j].up].down=a[j].down;
a[a[j].down].up=a[j].up;
// assert(a[j].y != col);
--cnt[a[j].c];
}
}
} // to uncover some column
void uncover(int col){
// 恢复第 col 列
a[a[col].left].right=col;
a[a[col].right].left=col;
// 恢复所有在第 col 列有元素的行
for(int i=a[col].down; i!=col; i=a[i].down){
for(int j=a[i].right; j!=i; j=a[j].right){
a[a[j].up].down=j;
a[a[j].down].up=j;
++cnt[a[j].c];
}
}
} void dance(int & res, int lev){
if(lev>=res) return; // 剪枝
int mi = INT_MAX, c=0; for(int i=a[0].right; i; i=a[i].right)
if(cnt[i] < mi){
mi = cnt[i];
c = i;
} if(c==0){
res=lev;
return;
};
// a[i].y == i holds.
cover(c);
// 枚举在第 i 列有元素的行
for(int i=a[c].down; i!=c; i=a[i].down){
// to choose some line
// res.push_back(a[i].x);
for(int j=a[i].right; j!=i; j=a[j].right){ // 覆盖在第 i 行有元素的列
cover(a[j].c);
}
// if(dance(res)) return true;
dance(res, lev+1); // res.pop_back();
for(int j=a[i].left; j!=i; j=a[j].left){
uncover(a[j].c);
}
}
uncover(c);
return;
}
// constructor
};

Dancing Links 的原理可以参考 hihoCoder Week #101 和 Donold Knuth 的论文

Dancing Links 的实现,网上的模板大都是 Knuth 论文里的伪代码风格: L[], R[], U[], D[], C[]四个数组。 构造双向十字循环链表的方法是定义 void link(int r, int c) 函数,每个插入一个元素(0-1矩阵中的某个1)。我认为这种写法是比较好的,简洁又灵活。我的实现与常见的写法略有不同,其差别无关紧要。

实现要点

  • 若不要求输出具体方案,则无需存储每个元素的列。若需要输出具体方案,则往往涉及从0-1矩阵的行编号到对应的决策的解码(decode)操作。
  • 选择0-1矩阵的某一行这一操作对应到对双向十字循环链表的操作中即“覆盖”(cover())这一行所能覆盖的那些列。
  • 根据具体题目修改 dance() 函数。

Dancing Links 模板的更多相关文章

  1. HUST 1017 - Exact cover (Dancing Links 模板题)

    1017 - Exact cover 时间限制:15秒 内存限制:128兆 自定评测 5584 次提交 2975 次通过 题目描述 There is an N*M matrix with only 0 ...

  2. ZOJ 3209 Treasure Map (Dancing Links)

    Treasure Map Time Limit: 2 Seconds      Memory Limit: 32768 KB Your boss once had got many copies of ...

  3. HUST1017(KB3-A Dancing links)

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

  4. [ACM] HUST 1017 Exact cover (Dancing Links,DLX模板题)

    DESCRIPTION There is an N*M matrix with only 0s and 1s, (1 <= N,M <= 1000). An exact cover is ...

  5. hust 1017 dancing links 精确覆盖模板题

    最基础的dancing links的精确覆盖题目 #include <iostream> #include <cstring> #include <cstdio> ...

  6. Dancing Links and Exact Cover

    1. Exact Cover Problem DLX是用来解决精确覆盖问题行之有效的算法. 在讲解DLX之前,我们先了解一下什么是精确覆盖问题(Exact Cover Problem)? 1.1 Po ...

  7. Dancing Links初学记

    记得原来备战OI的时候,WCX大神就研究过Dancing Links算法并写了一篇blog.后来我还写了个搜索策略的小文章( http://www.cnblogs.com/pdev/p/3952279 ...

  8. POJ 3074 Sudoku (Dancing Links)

    传送门:http://poj.org/problem?id=3074 DLX 数独的9*9的模板题. 具体建模详见下面这篇论文.其中9*9的数独怎么转化到精确覆盖问题,以及相关矩阵行列的定义都在下文中 ...

  9. poj 3074 Sudoku(Dancing Links)

    Sudoku Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 8152   Accepted: 2862 Descriptio ...

随机推荐

  1. 2017-3-7-lint82single-number

    2017-3-7-lint82single-number 在河之洲 算法 小书匠 problem 82single-number/ solution int singleNumber(vector&l ...

  2. ping请求超(iPV4)

    ping请求超(iPV4) arp –a(查看局域网全部IP) cmd 管理员运行 netsh winsock reset(重置Winsock目录借以恢复网络) etsh int ip reset r ...

  3. k8s1.13.0二进制部署-ETCD集群(一)

    Kubernetes集群中主要存在两种类型的节点:master.minion节点. Minion节点为运行 Docker容器的节点,负责和节点上运行的 Docker 进行交互,并且提供了代理功能.Ma ...

  4. python之函数名的应用

    1. 函数名是一个特殊的变量 例题 例题1: a = 1 b = 2 c = a + b print(c) # 输出结果 3 # 总结 # 变量是否可以进行相加或者拼接操作是又后面指向的值来决定的,指 ...

  5. C10 C语言数据结构

    目录 枚举 结构体 共用体 枚举 enum enum枚举是 C 语言中的一种基本数据类型,它可以让数据更简洁,更易读. 枚举语法定义格式为: enum 枚举名 {枚举元素1,枚举元素2,……}; 枚举 ...

  6. 使用Electron开发PC客户端

    最近公司要求开发一个PC客户端,要求不能使用.NET开发(为了不让用户安装.net framework),所以就选择了Electron(随口听别人说了一句,之前从来没有接触过).目前项目要完毕了,所以 ...

  7. 空类生成对象输出的结果是什么? toString()输出 覆写Object toString()方法输出的结果是什么

    空类生成对象输出的结果是什么? 输出的是对象在内存空间地址的哈希值 com.swift.P@1db9742 空类生成对象toString()输出的结果是什么? 输出的是对象在内存空间地址的哈希值的字符 ...

  8. python特殊字符转义符号表示

  9. Vue之数据传递

    基础:vue的响应式规则 简单的props更新 父组件 <template> <div> <block-a :out-data="x">< ...

  10. Golang map并发 读写锁

    golang并发 一:只有写操作 var ( count int l = sync.Mutex{} m = make(map[int]int) ) //全局变量并发写 导致计数错误 func vari ...