UVA1602 Lattice Animals 搜索+剪枝

题目大意

  给出一个$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;
}

  

 

转载于:https://www.cnblogs.com/headboy2002/p/9631719.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/365766.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python怎么打开程序管理器_Python 进程管理工具 Supervisor 使用教程

因为我的个人网站 restran.net 已经启用&#xff0c;博客园的内容已经不再更新。请访问我的个人网站获取这篇文章的最新内容&#xff0c;Python 进程管理工具 Supervisor 使用教程 Supervisor 是基于 Python 的进程管理工具&#xff0c;只能运行在 Unix-Like 的系统上&#xff…

ft服务器设置传输协议,ft服务器设置成主动模式

ft服务器设置成主动模式 内容精选换一换如果您选择使用SFS Turbo实现文件共享存储&#xff0c;此章节操作可跳过&#xff0c;您可以参见《SAP HANA用户指南》中的“格式化磁盘”章节&#xff0c;挂载Backup卷。NFS Server磁盘需要格式化&#xff0c;并挂载到相应的目录后&#…

mysql text字段导出_Mysql数据库的各种命令:

一、连接MYSQL格式&#xff1a; mysql -h主机地址 -u用户名 -p用户密码1、连接到本机上的MYSQL。 首先打开DOS窗口&#xff0c;然后进入目录mysqlbin&#xff0c;再键入命令mysql -u root -p&#xff0c;回车后提示你输密码. 注意用户名前可以有空格也可以没有空格&#xff0c;…

运行,JUnit! 跑!!!

JUnit与JavaScript和SVN一起是程序员经常开始使用的一些技术&#xff0c;甚至没有读过一篇博客文章&#xff0c;更不用说一本书了。 也许这是一件好事&#xff0c;因为它们看起来足够简单且易于理解&#xff0c;因此我们无需任何手册即可立即使用它们&#xff0c;但这也意味着它…

css3图形绘制

以下几个例子主要是运用了css3中border、bordr-radius、transform、伪元素等属性来完成的&#xff0c;我们先了解下它们的基本原理。 border&#xff1a;简单的来说border语法主要包含&#xff08;border-width、border-style、border-color&#xff09;三个属性。 „ border-t…

vueh5调用摄像头拍照_潜望式拍照5G手机盘点:售价相差数千元 究竟怎么选?

【dogkeji-科技犬】最近很多网友询问科技犬&#xff0c;目前支持50X潜望式长焦手机都有哪些&#xff0c;可否进行相关手机的推荐&#xff0c;今天就应大家的需求来盘点一下&#xff0c;给各位网友一些参考。推荐一&#xff0c;三星 Galaxy S20 U三星Galaxy S20 5G系列不仅搭载了…

09 事件对象

上篇介绍完我们js的事件流的概念之后&#xff0c;相信大家对事件流也有所了解了。那么接下来我们看一下jquery的事件操作。 在说jquery的每个事件之前&#xff0c;我们先来看一下事件对象 事件对象 Event 对象代表事件的状态&#xff0c;比如事件在其中发生的元素、键盘按键的状…

使用Struts2,Hibernate和MySQL创建个人MusicManager Web应用程序的研讨会

概述&#xff1a; 在本研讨会教程中&#xff0c;我们将使用Struts 2&#xff0c;Hibernate和MySQL数据库开发一个个人音乐管理器应用程序。 该Web应用程序可用于将您的音乐收藏添加到数据库中。 我们将显示用于添加唱片的表格&#xff0c;并在下面显示所有音乐收藏。 通过单击“…

链表快速排序python_Python一行代码实现快速排序的方法

今天将单独为大家介绍一下快速排序&#xff01; 一、算法介绍 排序算法&#xff08;Sorting algorithm&#xff09;是计算机科学最古老、最基本的课题之一。要想成为合格的程序员&#xff0c;就必须理解和掌握各种排序算法。其中"快速排序"&#xff08;Quicksort&…

自定义滚动条样式

啥都不说先看图: 注: 只适合chrom,不适用IE和fireFox 下面展示代码: 1 <html lang"en">2 <head>3 <meta charset"UTF-8">4 <title>CSS3自定义滚动条-轩枫阁</title>5 <style>6 header7 {8 font-family: …

rust为什么显示不了国服_Rust编程语言初探

静态、强类型而又不带垃圾收集的编程语言领域内&#xff0c;很久没有新加入者参与竞争了&#xff0c;大概大部分开发者认为传统的C/C的思路已经不太适合新时代的编程需求&#xff0c;即便有Ken Tompson这样的大神参与设计的golang也采用了GC的思路来设计其新一代的语言&#xf…

wps表格粗线和细线区别_详解论文中的表格技术

今天我们主要学习的技能如下&#xff1a;• 怎样用word做论文要求的三线表• 三线表中辅助线的断开• 表格或者图片自动编号1. 先普及一下&#xff0c;论文中的三线表吧。三线表以其形式简洁、功能分明、阅读方便而在科技论文中被推荐使用。三线表通常只有3条线&#xff0c;即顶…

如何自定义CSS滚动条的样式?

欢迎大家前往腾讯云 社区&#xff0c;获取更多腾讯海量技术实践干货哦~ 本文由前端林子发表 本文会介绍CSS滚动条选择器&#xff0c;并在demo中展示如何在Webkit内核浏览器和IE浏览器中&#xff0c;自定义一个横向以及一个纵向的滚动条。 0.需求 有的时候我们不想使用浏览器默…

RabbitMQ基础知识

RabbitMQ基础知识 一、背景RabbitMQ是一个由erlang开发的AMQP&#xff08;Advanced Message Queue &#xff09;的开源实现。AMQP 的出现其实也是应了广大人民群众的需求&#xff0c;虽然在同步消息通讯的世界里有很多公开标准&#xff08;如 COBAR的 IIOP &#xff0c;或者是 …

iview 级联选择组件_使用 element-ui 级联插件遇到的坑

需求描述【省市区三级联动】组件&#xff1a;Cascader 级联选择器后端需要所选中的地区的名字&#xff0c;如&#xff1a;[北京市, 北京市, 东城区]获取后端省市区具体列表的接口返回数据&#xff1a;// 省 - 参数1 [{value: 1,label: 北京市},... ] // 市 - 参数2 [{value: 1,…

深入理解CPU和异构计算芯片GPU/FPGA/ASIC (上篇)

王玉伟&#xff0c;腾讯TEG架构平台部平台开发中心基础研发组资深工程师&#xff0c;专注于为数据中心提供高效的异构加速云解决方案。目前&#xff0c;FPGA已在腾讯海量图片处理以及检测领域已规模上线。 随着互联网用户的快速增长&#xff0c;数据体量的急剧膨胀&#xff0c;…

jenkins-基础配置

一&#xff0c;配置远程连接服务器 系统管理 --> 系统设置 SSH remote hosts 二&#xff0c;设置docke的URL&#xff08;设置jenkins构建镜像时候所连接的docker url &#xff0c;参考 docker开启远程访问https://www.cnblogs.com/galsnag/articles/10069709.html&#xf…

JSF:直接从页面将参数传递给JSF操作方法,这是JavaEE 6+的一个不错的功能

Java企业版JavaEE 6中提供的JSF 2的一项不错的功能是&#xff0c;您可以将参数传递给任何操作组件&#xff08;例如commandButton或commandLink组件&#xff09;的操作方法。 基于此&#xff0c;您可以最大程度地减少托管bean中的方法数量。 另外&#xff0c;为了最小化在bea…

海洋主题绘画_深圳举办风帆时代海洋绘画作品展,展出作品600余件

12月12日&#xff0c;第七届《风帆时代海洋绘画作品展》在位于蛇口邮轮中心3楼的深圳大学海洋文化科普教育基地举行开幕仪式。该项目得到深圳市宣传文化事业专项基金支持&#xff0c;由深圳大学海洋艺术研究中心主办&#xff0c;深圳市海洋文化艺术研究会承办。作为开幕式重要环…

不要被约束的意思_不要再奢望你会变得自律了丨“他律”比“自律”更重要

高三寒假和同学打赌一个假期做完400套卷子。否则给他1000元。。。然后每天早上六点晚上12点&#xff0c;春节也没过&#xff0c;最后做完了卷子&#xff0c;我也完成了自己的梦想&#xff01;&#xff01;&#xff01;然而上面这个大神不是我&#xff0c;是我引用的一颗真实栗子…