Dancing Links用来解决如下精确匹配的问题:

选择若干行使得每一列恰好有一个1。Dancing Links通过对非零元素建立双向十字循环链表。上面的例子建立的链表如下所示:

计算的时候使用搜索的策略。每次选出1最少的一列,比如c,然后选择这一列中的某一行,比如r,(r,c)=1,然后r中所有1所在的列,那些其他行这些列有1的都删掉(这些行不会在r算入答案后也在答案里,否则就有某些列多于一个1出现)。然后这就变成一个规模更小的问题,继续搜索。无解时要回溯。

 class CDancingLinks
{
protected:
struct DancingLinksNode
{
DancingLinksNode* left;
DancingLinksNode* right;
DancingLinksNode* down;
DancingLinksNode* up;
int col;
int row;
}; typedef DancingLinksNode Node; int *m_columnEleNumbers;
int m_colNumber;
int m_rowNumber;
Node* m_pool;
Node** m_head;
int m_curUsePoolIndex; void _Remove(Node* cur)
{
--m_columnEleNumbers[cur->col];
for(Node* p=cur->down;p!=cur;p=p->down)
{
p->left->right=p->right;
p->right->left=p->left;
}
} void _Resume(Node* cur)
{
++m_columnEleNumbers[cur->col];
for(Node* p=cur->up;p!=cur;p=p->up)
{
p->left->right=p;
p->right->left=p;
}
} bool _SearchSolution(const int depth,std::vector<int> &solution)
{
Node* p=_GetNode();
if(p->left==p) return true; int Min=m_rowNumber+;
int MinColumnIndex=;
for(Node* q=p->left;q!=p;q=q->left)
{
if(m_columnEleNumbers[q->col]<Min)
{
Min=m_columnEleNumbers[q->col];
MinColumnIndex=q->col;
}
} for(Node* q=_GetNode(MinColumnIndex)->down;q!=_GetNode(MinColumnIndex);q=q->down)
{
_Remove(q);
solution.push_back(q->row);
for(Node* rr=q->right;rr!=q;rr=rr->right) _Remove(rr);
if(_SearchSolution(depth+,solution)) return true;
for(Node* rr=q->left;rr!=q;rr=rr->left) _Resume(rr);
solution.pop_back();
_Resume(q);
} return false;
} Node* _GetNode(int id) { return m_pool+id; } void _ReleaseMemory()
{
if(m_columnEleNumbers)
{
delete[] m_columnEleNumbers;
m_columnEleNumbers=nullptr;
} if(m_pool)
{
delete[] m_pool;
m_pool=nullptr;
}
if(m_head)
{
delete[] m_head;
m_head=nullptr;
}
} public: CDancingLinks():m_colNumber(-),m_rowNumber(-),
m_columnEleNumbers(nullptr),m_pool(nullptr),m_head(nullptr) {} /***
列下标为[1,Column]
***/
CDancingLinks(const int Column,const int Row):
m_columnEleNumbers(nullptr),m_pool(nullptr),m_head(nullptr)
{
SetSize(Column,Row);
} /***
列下标为[1,Column]
***/
void SetSize(const int Column,const int Row)
{
m_colNumber=Column;
m_rowNumber=Row; _ReleaseMemory(); m_columnEleNumbers=new int[m_colNumber+];
m_pool=new Node[m_colNumber*(m_rowNumber+)+];
m_head=new Node*[m_rowNumber+];
Clear();
} void Clear()
{
for(int i=;i<=m_colNumber;++i)
{
Node* cur=_GetNode(i);
cur->left=((i==m_colNumber)?_GetNode():_GetNode(i+));
cur->right=((==i)?_GetNode(m_colNumber):_GetNode(i-));
m_columnEleNumbers[i]=; cur->up=cur->down=_GetNode(i);
cur->col=i;
cur->row=;
}
for(int i=;i<=m_rowNumber;++i) m_head[i]=NULL;
m_curUsePoolIndex=m_colNumber+;
} ~CDancingLinks()
{
_ReleaseMemory();
} void AddElement(const int row,const int col)
{ Node* cur=m_pool+(m_curUsePoolIndex++); cur->up=_GetNode(col);
cur->down=_GetNode(col)->down;
m_pool[col].down->up=cur;
m_pool[col].down=cur; if(m_head[row]==NULL)
{
m_head[row]=cur->left=cur->right=cur;
}
else
{
cur->left=m_head[row]->left;
cur->right=m_head[row];
m_head[row]->left->right=cur;
m_head[row]->left=cur;
}
++m_columnEleNumbers[col];
cur->col=col;
cur->row=row;
} bool GetSolution(std::vector<int> &Solution)
{
return _SearchSolution(,Solution);
}
};

Dancing Links的更多相关文章

  1. Dancing Links and Exact Cover

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

  2. 跳跃的舞者,舞蹈链(Dancing Links)算法——求解精确覆盖问题

    精确覆盖问题的定义:给定一个由0-1组成的矩阵,是否能找到一个行的集合,使得集合中每一列都恰好包含一个1 例如:如下的矩阵 就包含了这样一个集合(第1.4.5行) 如何利用给定的矩阵求出相应的行的集合 ...

  3. ZOJ 3209 Treasure Map (Dancing Links)

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

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

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

  5. Dancing Links初学记

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

  6. 【转】Dancing Links题集

    转自:http://blog.csdn.net/shahdza/article/details/7986037 POJ3740 Easy Finding [精确覆盖基础题]HUST1017 Exact ...

  7. 【转】Dancing Links精确覆盖问题

    原文链接:http://sqybi.com/works/dlxcn/ (只转载过来一部分,全文请看原文,感觉讲得很好~)正文    精确覆盖问题    解决精确覆盖问题    舞蹈步骤    效率分析 ...

  8. POJ 3074 Sudoku (Dancing Links)

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

  9. HDU5046 Airport dancing links 重复覆盖+二分

    这一道题和HDU2295是一样 是一个dancing links重复覆盖解决最小支配集的问题 在给定长度下求一个最小支配集,只要小于k就行 然后就是二分答案,每次求最小支配集 只不过HDU2295是浮 ...

随机推荐

  1. ASP.NET MVC4中的bundles特性引发服务器拒绝访问(403错误)

    在ASP.NET MVC4中微软引入了bundles特性,这个特性可以将服务器端的多个Javascript或多个css文件捆绑在一起作为一个单一的URL地址供客户端浏览器调用,从而减少了页面上Http ...

  2. SQL SERVER: 合并相关操作(Union,Except,Intersect) - 转载

    SQL Server 中对于结果集有几个处理,值得讲解一下 1. 并集(union,Union all) 这个很简单,是把两个结果集水平合并起来.例如 SELECT * FROM A UNION SE ...

  3. JavaSPI机制学习笔记

    最近在阅读框架源代码时,常常看到 SPI 的子包, 忍不住查了下: Service Provider Interface : 服务提供接口. JavaSPI 实际上是“基于接口的编程+策略模式+配置文 ...

  4. linux curl用法详解

    linux ‍‍curl用法详解 ‍‍curl的应用方式,一是可以直接通过命令行工具,另一种是利用libcurl库做上层的开发.本篇主要总结一下命令行工具的http相关的应用, 尤其是http下载方面 ...

  5. 分析Linux内核创建一个新进程的过程【转】

    转自:http://www.cnblogs.com/MarkWoo/p/4420588.html 前言说明 本篇为网易云课堂Linux内核分析课程的第六周作业,本次作业我们将具体来分析fork系统调用 ...

  6. 怎么使用 Laravel 的服务容器来优化读写数据库中的 options关键词

    其中我们可以最方便地利用的一个特性就是 Laravel 的服务容器了.在这里我不多赘述 Service Container 是个啥,想了解的可以自行搜索.不想了解的就只要大致知道它是个可以 绑定/取出 ...

  7. Window驱动开发

    驱动开发 参考文章: Windbg+Vmware驱动调试 http://blog.csdn.net/xuepiaosong/article/details/8236702 驱动调试攻略(WinDbg) ...

  8. recycleview中使用checkbox导致的重复选中问题

    参考博文:http://www.myexception.cn/mobile/1852852.html 在使用RecycleView做仿微信图片选择器,其中条目中使用了checkbox,在选中时由于ho ...

  9. 浮点数转换为人名币读法字符串(JAVA)

    /*<java疯狂讲义>浮点数转换为人名币读法字符串这个用例,感觉没有考虑零的情况*/ import java.util.Arrays; public class Num2Rmb { pr ...

  10. [C++][语言语法]标准C++中的string类的用法总结

    转自:http://www.cnblogs.com/xFreedom/archive/2011/05/16/2048037.html 要想使用标准C++中string类,必须要包含 #include ...