Easy Finding
这道题运用二进制DFS是比较好的方法,当然还可以用dancing links,这个比较高级,先把我看别人的二进制思路然后写的代码贴上来。
当然本题是体现不出DLX的高级之处的。用位运算和DLX的速度应该是相当
/*
ID: sdj22251
PROG: calfflac
LANG: C++
*/
#include <iostream>
#include <vector>
#include <list>
#include <map>
#include <set>
#include <deque>
#include <queue>
#include <stack>
#include <bitset>
#include <algorithm>
#include <functional>
#include <numeric>
#include <utility>
#include <sstream>
#include <iomanip>
#include <cstdio>
#include <cmath>
#include <cstdlib>
#include <cctype>
#include <string>
#include <cstring>
#include <cmath>
#include <ctime>
#define MAX 2000000000
#define LOCA
#define PI acos(-1.0)
using namespace std;
int n, m, bit[18], a[18][303], cnt[303];
bool flag;
void dfs(int num, int in, int out) //in是已经使用的,out是排除了的,num是深度
{
int i, j;
if(flag) return; // 找到了就一直return
if(num >= n) {flag = true; return;} //找到了
if(cnt[num] == (cnt[num] & out)) return; //如果深搜的目标都在排除的序列中,就return
int count = 0;
for(i = 0; i < m && count < 2; i++)
if(cnt[num] & in & bit[i]) {j = i; count++;} //如果目标已经使用并且相应位置是1
if(count == 1) dfs(num + 1, in, out | cnt[num] ^ bit[j]); //则接着搜索时排除除了该1位置以外的1
else if (count == 0) //还没被使用过
{
for(i = 0; i < m && !flag; i++)
if((cnt[num] & bit[i]) && !(out & bit[i])) // 相应位置是1 并且还没被排除过
dfs(num + 1, in | bit[i], out | cnt[num] ^ bit[i]);
}
}
int main()
{
#ifdef LOCAL
freopen("ride.in","r",stdin);
freopen("ride.out","w",stdout);
#endif
int i, j;
bit[0] = 1;
for(i = 1; i < 18; i++) //将不同位置的1预存起来
bit[i] = bit[i - 1] << 1;
while(scanf("%d%d", &m, &n) != EOF)
{
memset(cnt, 0, sizeof(cnt));
for(i = 0; i < m; i++)
{
for(j = 0; j < n; j++)
{
scanf("%d", &a[i][j]);
if(a[i][j])
cnt[j] |= bit[i]; //如果这一个点是1,就将1及其位置存进二进制数组
}
}
flag = false;
for(i = 0; i < n; i++) //筛掉全是0这个情况
if(!cnt[i]) break;
if(i >= n) dfs(0, 0, 0);
if(flag) puts("Yes, I found it");
else puts("It is impossible");
}
return 0;
}
2012年1月
然后就是传说中的DLX了。
看了网上很多的资料,只是建图的时候比较麻烦吧
每一列先建立一个列头,然后往某一列中插入元素时只需要插到列头下面就行,因为是循环链表插到哪里都一样的
而行呢,不一定要行头,但是最好能把每一行的第一个1的编号记录下来,每次插入到它之后就行了。
/*
ID: sdj22251
PROG: inflate
LANG: C++
*/
#include <iostream>
#include <vector>
#include <list>
#include <map>
#include <set>
#include <deque>
#include <queue>
#include <stack>
#include <bitset>
#include <algorithm>
#include <functional>
#include <numeric>
#include <utility>
#include <sstream>
#include <iomanip>
#include <cstdio>
#include <cmath>
#include <cstdlib>
#include <cctype>
#include <string>
#include <cstring>
#include <cmath>
#include <ctime>
#define MAXN 5000
#define INF 1000000000
using namespace std;
int L[MAXN], R[MAXN], C[MAXN], S[MAXN], U[MAXN], D[MAXN], H[MAXN];//H数组用来存储每一行的第一个1的编号,C数组用来存列号,S数组用来存每一列中1的个数
int n, m, cnt, head;//head表示列头,一般取0,cnt用来记录编号
void link(int r, int c)
{
S[c]++;
C[cnt] = c;
U[cnt] = c;
D[cnt] = D[c];
U[D[c]] = cnt;
D[c] = cnt;
if(H[r] == -1)
{
H[r] = cnt;
L[cnt] = R[cnt] = cnt;
}
else
{
L[cnt] = H[r];
R[cnt] = R[H[r]];
L[R[H[r]]] = cnt;
R[H[r]] = cnt;
}
cnt++;
}
void init()
{
cnt = 0;
head = 0;
for(int i = 0; i <= m; i++)
{
S[i] = 0;
D[i] = U[i] = i;
R[i] = (i + 1) % (m + 1);
L[i] = (i + m) % (m + 1);
cnt++;
}
memset(H, -1, sizeof(H));
}
void readdata()
{
char t;
getchar();
for(int i = 1; i <= n; i++)
for(int j = 1; j <= m; j++)
{
t = getchar();
getchar();
if(t == '1') link(i, j);
}
}
void removes(int c)
{
L[R[c]] = L[c];
R[L[c]] = R[c];
for(int i = D[c]; i != c; i = D[i])
for(int j = R[i]; j != i; j = R[j])
{
U[D[j]] = U[j];
D[U[j]] = D[j];
S[C[j]]--;
}
}
void resumes(int c)
{
for(int i = U[c]; i != c; i = U[i])
for(int j = L[i]; j != i; j = L[j])
{
U[D[j]] = j;
D[U[j]] = j;
S[C[j]]++;
}
L[R[c]] = c;
R[L[c]] = c;
}
bool dfs(int k)
{
if(R[head] == head) return true;
int s = INF, c;
for(int i = R[head]; i != head; i = R[i])
if(s > S[i])
{
s = S[i];
c = i;
}
removes(c);
for(int i = U[c]; i != c; i = U[i])
{
for(int j = R[i]; j != i; j = R[j])
removes(C[j]);
if(dfs(k + 1)) return true;
for(int j = L[i]; j != i; j = L[j])
resumes(C[j]);
}
resumes(c);
return false;
}
int main()
{
while(scanf("%d%d", &n, &m) != EOF)
{
init();
readdata();
if(dfs(0)) puts("Yes, I found it");
else puts("It is impossible");
}
return 0;
}
分享到:
相关推荐
北大POJ3009-Curling 2.0【DFS+Vector+回溯+剪枝】 解题报告+AC代码
北大POJ3733-Changing Digits【DFS+强剪枝】 解题报告+AC代码
北大POJ3373-Changing Digits【DFS+强剪枝】 解题报告+AC代码
POJ-2870 Light Up + DFS(1级DFS+1级DFS) + Python - 思维导图 链接:https://blog.csdn.net/xxdragon126/article/details/122095922?spm=1001.2014.3001.5501
北大POJ1691-Painting A Board 【拓扑+DFS】 解题报告+AC代码
北大POJ3026-Borg Maze【BFS+Prim】 解题报告+AC代码
北大POJ初级-所有题目AC代码+解题报告
POJ2186-Popular Cows 【Tarjan+极大强连通分量+缩点】 解题报告+AC代码 http://hi.csdn.net/!s/BGDH68 附:我所有的POJ解题报告链接 . http://blog.csdn.net/lyy289065406/article/details/6642573
北大POJ1020-Anniversary Cake 解题报告+AC代码
1.把自己的污水排到河里V 2.把自己的污水送到右边> 3.把自己的污水送到左边
北大POJ初级-简单搜索 解题报告+AC代码
北大POJ初级题-数据结构:解题报告+AC代码
poj 3376 Finding Palindromes.md
北大POJ2187-Beauty Contest 解题报告+AC代码
北大POJ3252-Round Numbers 解题报告+AC代码
北大POJ1416-Shredding Company 解题报告+AC代码
北大POJ1019-Number Sequence 解题报告+AC代码
北大POJ3126-Prime Path 解题报告+AC代码
北大POJ1472-Instant Complexity 解题报告+AC代码
北大POJ1018-Communication System 解题报告+AC代码