poj-3740 Easy Finding
2015-11-11 08:03
381 查看
题意:
给出一个n*m的01矩阵,选择其中的一些行,来精确覆盖每一列;
只需要输出是否存在解即可;
n<=16,m<=300;
题解:
DLX裸题,利用双向十字链表优化搜索中的回溯问题;
因为每一列上都只能有且仅有一个1,所以如果某一列上已经有了1,那么这一列上有1的其他行也可以被删除;
根据这个思想是我们有了一个很厉害的剪枝条件,但是如果直接在矩阵中删除速度太慢,要求空间太多;
所以就有了这种支持快速删除与恢复的数据结构——双向链表的优化;
具体实现上,我觉得还是指针比较靠谱,因为所以指针都是循环的所以不存在访问NULL的问题;
代码:
#include<stdio.h>
#include<string.h>
#include<algorithm>
#define N 20
#define M 310
using namespace std;
struct node
{
node *l,*r,*u,*d;
int belong;
}buf[N*M],*head,*c[M];
int tot,cnt[M];
bool map
[M];
node *newnode()
{
return &buf[++tot];
}
void Build(int n,int m)
{
tot=0;
head=newnode();
for(int i=1;i<=m;i++)
c[i]=newnode();
for(int i=1;i<=m;i++)
{
c[i]->l=c[i-1],c[i]->r=c[i+1];
c[i]->u=c[i],c[i]->d=c[i];
}
c[1]->l=head,c[m]->r=head;
head->r=c[1],head->l=c[m];
for(int i=1;i<=n;i++)
{
node *t=NULL;
for(int j=1;j<=m;j++)
{
if(map[i][j])
{
node *now=newnode();
if(t==NULL)
now->l=now->r=now;
else
now->l=t,now->r=t->r,
t->r=now,now->r->l=now;
now->belong=j;
now->u=c[j]->u;
now->d=c[j];
c[j]->u->d=now;
c[j]->u=now;
t=now;
}
}
}
}
void remove(node *x)
{
x->l->r=x->r;
x->r->l=x->l;
for(node *t=x->d;t!=x;t=t->d)
{
for(node *k=t->r;k!=t;k=k->r)
{
k->u->d=k->d;
k->d->u=k->u;
}
}
}
void resume(node *x)
{
x->l->r=x;
x->r->l=x;
for(node *t=x->d;t!=x;t=t->d)
{
for(node *k=t->r;k!=t;k=k->r)
{
k->u->d=k;
k->d->u=k;
}
}
}
bool dfs()
{
node *t=head->r;
if(t==head) return 1;
for(node *i=t->d;i!=t;i=i->d)
{
remove(t);
for(node *j=i->r;j!=i;j=j->r)
{
remove(c[j->belong]);
}
if(dfs())
return 1;
for(node *j=i->r;j!=i;j=j->r)
{
resume(c[j->belong]);
}
resume(t);
}
return 0;
}
int main()
{
int n,m,i,j,k;
while(scanf("%d%d",&n,&m)!=EOF)
{
for(i=1;i<=n;i++)
{
for(j=1;j<=m;j++)
{
scanf("%d",&map[i][j]);
}
}
Build(n,m);
if(dfs())
puts("Yes, I found it");
else
puts("It is impossible");
}
return 0;
}
给出一个n*m的01矩阵,选择其中的一些行,来精确覆盖每一列;
只需要输出是否存在解即可;
n<=16,m<=300;
题解:
DLX裸题,利用双向十字链表优化搜索中的回溯问题;
因为每一列上都只能有且仅有一个1,所以如果某一列上已经有了1,那么这一列上有1的其他行也可以被删除;
根据这个思想是我们有了一个很厉害的剪枝条件,但是如果直接在矩阵中删除速度太慢,要求空间太多;
所以就有了这种支持快速删除与恢复的数据结构——双向链表的优化;
具体实现上,我觉得还是指针比较靠谱,因为所以指针都是循环的所以不存在访问NULL的问题;
代码:
#include<stdio.h>
#include<string.h>
#include<algorithm>
#define N 20
#define M 310
using namespace std;
struct node
{
node *l,*r,*u,*d;
int belong;
}buf[N*M],*head,*c[M];
int tot,cnt[M];
bool map
[M];
node *newnode()
{
return &buf[++tot];
}
void Build(int n,int m)
{
tot=0;
head=newnode();
for(int i=1;i<=m;i++)
c[i]=newnode();
for(int i=1;i<=m;i++)
{
c[i]->l=c[i-1],c[i]->r=c[i+1];
c[i]->u=c[i],c[i]->d=c[i];
}
c[1]->l=head,c[m]->r=head;
head->r=c[1],head->l=c[m];
for(int i=1;i<=n;i++)
{
node *t=NULL;
for(int j=1;j<=m;j++)
{
if(map[i][j])
{
node *now=newnode();
if(t==NULL)
now->l=now->r=now;
else
now->l=t,now->r=t->r,
t->r=now,now->r->l=now;
now->belong=j;
now->u=c[j]->u;
now->d=c[j];
c[j]->u->d=now;
c[j]->u=now;
t=now;
}
}
}
}
void remove(node *x)
{
x->l->r=x->r;
x->r->l=x->l;
for(node *t=x->d;t!=x;t=t->d)
{
for(node *k=t->r;k!=t;k=k->r)
{
k->u->d=k->d;
k->d->u=k->u;
}
}
}
void resume(node *x)
{
x->l->r=x;
x->r->l=x;
for(node *t=x->d;t!=x;t=t->d)
{
for(node *k=t->r;k!=t;k=k->r)
{
k->u->d=k;
k->d->u=k;
}
}
}
bool dfs()
{
node *t=head->r;
if(t==head) return 1;
for(node *i=t->d;i!=t;i=i->d)
{
remove(t);
for(node *j=i->r;j!=i;j=j->r)
{
remove(c[j->belong]);
}
if(dfs())
return 1;
for(node *j=i->r;j!=i;j=j->r)
{
resume(c[j->belong]);
}
resume(t);
}
return 0;
}
int main()
{
int n,m,i,j,k;
while(scanf("%d%d",&n,&m)!=EOF)
{
for(i=1;i<=n;i++)
{
for(j=1;j<=m;j++)
{
scanf("%d",&map[i][j]);
}
}
Build(n,m);
if(dfs())
puts("Yes, I found it");
else
puts("It is impossible");
}
return 0;
}
相关文章推荐
- 搜狗百度360市值齐跌:搜索引擎们陷入集体焦虑?
- 本人即将筹备败家日志,敬请期待!
- IE:使用搜索助手
- C++深度优先搜索的实现方法
- 基于文本的搜索
- php实现搜索一维数组元素并删除二维数组对应元素的方法
- 使用Sphinx对索引进行搜索
- asp 多关键词搜索的简单实现方法
- C#使用foreach语句搜索数组元素的方法
- JavaScript中数组的排序、乱序和搜索实现代码
- C#编程实现Excel文档中搜索文本内容的方法及思路
- sqlserver中在指定数据库的所有表的所有列中搜索给定的值
- 可以用来搜索当前页面内容的js代码
- 全文搜索和替换
- javascript搜索自动提示功能的实现第1/3页
- mysql 模糊搜索的方法介绍
- C#搜索文字在文件及文件夹中出现位置的方法
- 基于ASP.NET的lucene.net全文搜索实现步骤
- 做个自己站内搜索引擎
- PHP查找与搜索数组元素方法总结