题目大意

  给出一个$w\times h$的网格,定义一个连通块为一个元素个数为$n$的方格的集合$A,\forall x\in A, \exists y\in A$,使得$x,y$有一条公共边。现要求一个元素个数极多的连通块的集合$K_N$,使得$\forall A,B\in K_n$,不存在一种由对称、旋转、平移组成的操作$f(A)$,使得$f(A)=B$。求$|K|$。$w\leq h\leq n\leq 10$。

题解

  这道题可把我恶心到吐了。。。

主体算法

错误算法1

  跟宝藏一题的思路一样,把两个相邻的块看作由一条边连接的两个节点,随后枚举生成树的个数$m$即可。最后输出的结果便是$\frac{m}{8}$(对称/2,旋转再/4)。

  这种做法错因在于不同的连通块可以对应不同的生成树,而且连通块可能会有轴对称或旋转对称等。以后遇到像这样复杂得连手推样例都不愿做的算法就不要再使用了。

错误算法2

  对于每一组数据,进行Dfs搜索。将Dfs搜索到的连接块用set记录,它的作用一是防止一个连通块被访问多次。二是防止等价的最终连通块使答案变多,实现是要将最终连通块本身以及经过对称旋转后的变化图形卡在网格的左上角。

  这种做法错在多组数据都要重算一遍,时间复杂度太高。

正确算法

  我们运用打表的思想,Bfs将$10\times 10$网格内的$K_n, n=1,2,\cdots 10$全部预处理出来保存在set<Block> Ans[15]中,给出$n,w,h$,则在Ans[n]中找到符合条件(注意:对于长径和宽径不同的连通块,不但要看看正常放能不能放入网格中,还要看看把它旋转90°后能不能放入)的连通块即可。

  怎么Bfs呢?Bfs的每一层按照连通块内格子的个数进行分类。Bfs的起点是一个只有格子(1, 1)的连通块。每一个Ans[n]内的连通块都是恰好卡到左上角的连通块。Ans[i]由Ans[i - 1]扩展得来,对于Ans[i - 1]中的每个连通块$A$,尝试找出两个相邻的网格$a, b, a\in A, b\notin A$,若它经过各种变换后的图形在Ans[i]中都没有出现,则将该连通块加入Ans[i]。

  问题来了,如何处理下列情况呢?

  如果我们就在网格的限制内扩展,这种扩展就不可能。所以我们可以尝试扩展到整个网格以外,然后卡到左上角的方法达到这个目的。因为每次只扩展一个格子,所以这种方法可行。

  不过这时我们不要忘了特判不合法的情况:

矩阵操作

方法一

  在每个连通块内维护一个set,维护每个在连通块内的网格的坐标。

  这样处理有些麻烦。

方法二

  每个连通块用二维数组存储,变换以整个网格的某个位置作为中心或轴。

  这个时候旋转90°操作就比较膈应人了。我们不可以以整个网格的中心作为轴,因为。。。待会,到底是旋转到哪儿去?中心是哪儿?  我们不如直接以(0, 0)作为中心旋转,将旋转过后的坐标用vector存储起来,然后将vector内的所有网格(row, col)加上(dRow, dCol)再存储起来。

  这种做法就是感觉有些不专业。。。

最终做法

  每个连通块用二维数组存储,操作在连通块所在矩形内操作。

  如图蓝色的矩形即为红色连通块对应的矩形。

  此时如何旋转90°呢?我们要用矩阵转置的方法。

然后这个程序就在50ms内通过了。

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <set>
#include <cassert>
using namespace std; #define UpdateMin(x, y) x = min(x, y)
#define UpdateMax(x, y) x = max(x, y) const int MAX_POS_ARG_SIZE = 15, INF = 0x3f3f3f3f, MINF = 0xcfcfcfcf;
const int TotRow = 10, TotCol = 10, TotNode = 10;
const int Dir[4][2] = { {0, 1}, {1, 0}, {0, -1}, {-1, 0} }; struct Block
{
bool A[MAX_POS_ARG_SIZE][MAX_POS_ARG_SIZE]; Block()
{
memset(A, 0, sizeof(A));
} bool operator < (const Block& a) const
{
return memcmp(A, a.A, sizeof(A)) == -1;
} bool operator == (const Block& a) const
{
return memcmp(A, a.A, sizeof(A)) == 0;
} Block operator = (const Block& a)
{
memcpy(A, a.A, sizeof(A));
return *this;
} void GetRect(int &minRow, int &maxRow, int &minCol, int &maxCol)//所在矩形
{
minRow = minCol = INF;
maxRow = maxCol = MINF;
for (int row = 0; row <= TotRow; row++)
for (int col = 0; col <= TotCol; col++)
if (A[row][col])
{
UpdateMin(minRow, row);
UpdateMax(maxRow, row);
UpdateMin(minCol, col);
UpdateMax(maxCol, col);
}
} void GetSq(int &minRow, int &maxRow, int &minCol, int &maxCol)//所在正方形
{
GetRect(minRow, maxRow, minCol, maxCol);
int eLen = max(maxRow - minRow + 1, maxCol - minCol + 1);
maxRow = minRow + eLen - 1;
maxCol = minCol + eLen - 1;
} Block GetFlipHor()
{
Block ans;
int minRow, maxRow, minCol, maxCol;
GetRect(minRow, maxRow, minCol, maxCol);
for (int row = minRow; row <= maxRow; row++)
for (int col1 = minCol, col2 = maxCol; col1 <= maxCol; col1++, col2--)
ans.A[row][col2] = A[row][col1];
return ans;
} Block GetRotate90()
{
Block ans;
int minRow, maxRow, minCol, maxCol;
GetRect(minRow, maxRow, minCol, maxCol);
for (int row = 1; row <= maxCol; row++)
for (int col = 1; col <= maxRow; col++)
ans.A[row][col] = A[maxRow -col +1][row];
return ans;
} Block GetNormal()
{
Block ans;
int minRow, maxRow, minCol, maxCol;
GetRect(minRow, maxRow, minCol, maxCol);
for (int row = minRow; row <= maxRow; row++)
for (int col = minCol; col <= maxCol; col++)
ans.A[row - minRow + 1][col - minCol + 1] = A[row][col];
return ans;
} bool Invalid(int totNode)
{
int minRow, maxRow, minCol, maxCol;
GetRect(minRow, maxRow, minCol, maxCol);
return maxRow - minRow + 1 > totNode || maxCol - minCol + 1 > totNode;
}
}; set<Block> Ans[MAX_POS_ARG_SIZE]; bool Exists_Rotate(set<Block> &ans, Block cur)
{
if (ans.count(cur))
return true;
for (int i = 1; i <= 3; i++)
{
cur = cur.GetRotate90();
if (ans.count(cur))
return true;
}
return false;
} bool Exists(set<Block> &ans, Block cur)
{
if (Exists_Rotate(ans, cur))
return true;
if (Exists_Rotate(ans, cur.GetFlipHor().GetNormal()))
return true;
return false;
} void MakeAnsList()
{
Block start;
start.A[1][1] = true;
Ans[1].insert(start);
for (int i = 2; i <= TotNode; i++)
{
for (set<Block>::iterator it = Ans[i - 1].begin(); it != Ans[i - 1].end(); it++)
{
for (int row = 1; row <= TotRow; row++)
for (int col = 1; col <= TotCol; col++)
{
if (it->A[row][col])
{
for (int k = 0; k < 4; k++)
{
int nextRow = row + Dir[k][0], nextCol = col + Dir[k][1];
if (nextRow < 0 || nextRow > TotRow || nextCol < 0 || nextCol > TotCol)
continue;
if (it->A[nextRow][nextCol])
continue;
Block next = *it;
next.A[nextRow][nextCol] = true;
if (next.Invalid(i))
continue;
next = next.GetNormal();
if (Exists(Ans[i], next))
continue;
Ans[i].insert(next);
}
}
}
}
}
} int GetAns(int totNode, int totRow, int totCol)
{
int ans = 0;
for (set<Block>::iterator it = Ans[totNode].begin(); it != Ans[totNode].end(); it++)
{
Block temp = *it;
int minRow, maxRow, minCol, maxCol;
temp.GetRect(minRow, maxRow, minCol, maxCol);
assert(minRow == 1 && minCol == 1);
if (maxRow <= totRow && maxCol <= totCol || maxCol <= totRow && maxRow <= totCol)
{
Block a = *it;
ans++;
}
}
return ans;
} int main()
{
MakeAnsList();
int totNode, totRow, totCol;
while (~scanf("%d%d%d", &totNode, &totRow, &totCol))
printf("%d\n", GetAns(totNode, totRow, totCol));
return 0;
}

  

UVA1602 Lattice Animals 搜索+剪枝的更多相关文章

  1. UVA-1602 Lattice Animals 搜索问题(打表+set)

    题目链接 https://vjudge.net/problem/UVA-1602 紫书的一道例题,跟之前的很多题目有很多不同. 本题不像是一般的dfs或bfs这样的搜索套路,而是另一种枚举思路. 题意 ...

  2. UVA1602 Lattice Animals 网格动物 (暴力,STL)

    多联骨牌的生成办法,维基上只找到固定的骨牌fix,而free的没有找到. 于是只好写个set判重的简单枚举了. 旋转的操作,可以在坐标轴上画个点,以原点为轴心,逆时针旋转90度,新的点的坐标为(-y, ...

  3. 【DFS】【打表】Lattice Animals

    [ZOJ2669]Lattice Animals Time Limit: 5 Seconds      Memory Limit: 32768 KB Lattice animal is a set o ...

  4. NOIP2015 斗地主(搜索+剪枝)

    4325: NOIP2015 斗地主 Time Limit: 30 Sec  Memory Limit: 1024 MBSubmit: 270  Solved: 192[Submit][Status] ...

  5. hdu 5469 Antonidas(树的分治+字符串hashOR搜索+剪枝)

    题目链接:hdu 5469 Antonidas 题意: 给你一颗树,每个节点有一个字符,现在给你一个字符串S,问你是否能在树上找到两个节点u,v,使得u到v的最短路径构成的字符串恰好为S. 题解: 这 ...

  6. hdu 5887 搜索+剪枝

    Herbs Gathering Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)T ...

  7. hdu 5113(2014北京—搜索+剪枝)

    题意:有N*M的棋盘,用K种颜色去染,要求相邻块不能同色.已知每种颜色要染的块数,问能不能染,如果能,输出任一种染法. 最开始dfs失败了- -,优先搜索一行,搜完后进入下一列,超时.本来以为搜索不行 ...

  8. luogu 1731 搜索剪枝好题

    搜索剪枝这个东西真的是骗分利器,然鹅我这方面菜的不行,所以搜索数学dp三方面是真的应该好好训练一下 一本通的确该认真的刷嗯 #include<bits/stdc++.h> using na ...

  9. 搜索+剪枝——POJ 1011 Sticks

    搜索+剪枝--POJ 1011 Sticks 博客分类: 算法 非常经典的搜索题目,第一次做还是暑假集训的时候,前天又把它翻了出来 本来是想找点手感的,不想在原先思路的基础上,竟把它做出来了而且还是0 ...

随机推荐

  1. 【C++】朝花夕拾——树(开篇)

    树 ===================我是分割线====================== 1. 定义: 一些结点的集合,集合可以为空.定义树的自然方式是递归的方法. 2. 相关概念: 根(ro ...

  2. HDU_1520_Anniversary party_树型dp

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1520 Anniversary party Time Limit: 2000/1000 MS (Java ...

  3. codeforces_304C_数学题

    C. Lucky Permutation Triple time limit per test 2 seconds memory limit per test 256 megabytes input ...

  4. HDU - 2044 - 一只小蜜蜂...(dp)

    题意: 如题 思路: 仔细观察图 1-4和3-6其实是一样的答案,那么所有的方案都可以相减,意思为全部转化为从1开始 剩下的就是观察规律,仔细观察5号,能到5号蜂房的只有3和4,3和4到5号蜂房只有一 ...

  5. [luogu4056 JSOI2009] 火星藏宝图 (贪心 dp)

    传送门 Solution 一个显然的贪心:选的点数越多越好.这个随便推推就知道了. 那么我们就贪心的从一列上挑最靠下的转移 直接转移不斜率优化复杂度\(O(nm)\),吸一口O2过了... Code ...

  6. <MyBatis>入门四 传入的参数处理

    1.单个参数 传入单个参数时,mapper文件中 #{}里可以写任意值 /** * 传入单个参数 */ Employee getEmpById(Integer id); <!--单个参数 #{} ...

  7. PAT 1122 Hamiltonian Cycle

    The "Hamilton cycle problem" is to find a simple cycle that contains every vertex in a gra ...

  8. 如何通过js在子页面调用父页面元素的click事件

    //获取父页面的某个元素var node = window.parent.document.getElementById("btnReturn");//调用该元素的Click事件 ...

  9. 【Codeforces 140C】New Year Snowmen

    [链接] 我是链接,点我呀:) [题意] 题意 [题解] 每次都选择剩余个数最多的3个不同数字组成一组. 优先消耗剩余个数多的数字 这样能尽量让剩余的数字总数比较多,从而更加可能得到更多的3个组合 [ ...

  10. 【Codeforces 1037D】Valid BFS?

    [链接] 我是链接,点我呀:) [题意] 让你判断一个序列是否可能为一个bfs的序列 [题解] 先dfs出来每一层有多少个点,以及每个点是属于哪一层的. 每一层的bfs如果有先后顺序的话,下一层的节点 ...