USACO Section 2.1 The Castle 解题报告
题目
题目描述
有一个城堡,城堡中有若干个房间,房间与房间之间用墙来进行分隔。现在我们需要统计这个城堡有多少个房间,并且还要找出最大的房间的面积是多少(一个单元格就代表一个单元面积)。城堡的主人现在想要在这些房间中通过打通一面墙来使两个房间合并,并且要求合并之后的房间是所有可能情况中最大的房间。我们需要计算的是,合并之后最大的房间是多大,需要打通的墙壁是哪一面。我们进行房间改造有一定的规则:从城堡的左下角开始往右上角遍历单元格,每个单元格只要考虑两面墙(如果有相应的墙壁),优先考虑朝向为北的墙壁(N
),之后再考虑朝向为东的墙壁(E
)。
问题输入
输入与平常有些不同,我们是每个单元格中有一个数字x
。现在我们有如下规定:
- 1:代表朝向为西的墙壁
- 2:代表朝向为北的墙壁
- 4:代表朝向为东的墙壁
- 8:代表朝向为南的墙壁
如果x
可以由1,2,4,8
这几个数中的某些数字相加得到,那就代表这个位置有相应的墙壁。城堡最大为50*50
的单元格组成。
问题输出
输出房间改造之前的房间数、最大的房间面积。然后输出房间改造之后的最大的房间面积、需要改造的墙壁。
样例输入
7 4
11 6 11 6 3 10 6
7 9 6 13 5 15 5
1 10 12 7 13 7 5
13 11 10 8 10 12 13
样例输出
5
9
16
4 1 E
解题思路
既然是要按照一定的规则输出最优的方案,我们肯定是需要遍历所有可能的情况。所以我最一开始就直接用宽搜找到所有的房间,并且统计每个房间的大小。之后我再枚举每一面墙壁,在枚举墙壁的时候思维比较混乱,因为之前统计房间我没有对房间进行编号,并且没有保存每个房间的面积。所以在枚举墙壁的时候我又用洪泛式的搜索来计算去掉这面墙壁之后可以增加多大的面积。最后的结果当然是超时啦......
所以后来看了下别人的思路提示,才想起来应该在统计房间大小的时候顺便对房间进行编号,而且还要对房间的大小进行记录,便于之后枚举墙壁的时候直接可以获得这些信息。
在解题的过程中我们可以发现墙壁的方向是1,2,4,8
,这是2的幂次方,所以我们可以用位运算的技巧来判断每个单元格的墙壁情况。这是一个解题技巧,可以让程序变得更加优雅。
解题代码
/*
ID: yinzong2
PROG: castle
LANG: C++11
*/
#define MARK
#include <iostream>
#include <map>
#include <queue>
#include <algorithm>
using namespace std;
const int maxn = 55;
int M, N;
// castle存地图,roomId用来保存每个点所属的房间编号
int castle[maxn][maxn], roomId[maxn][maxn];
map<int, int> roomSize; // 房间号映射到房间大小
int maxSize1, maxSize2;
int dirX[4] = {0, -1, 0, 1};
int dirY[4] = {-1, 0, 1, 0};
struct Room {
int x,y;
int value;
};
void findRoom(int x, int y, int roomNum) {
int rSize = 0;
queue<Room> Q;
while (!Q.empty())Q.pop();
Room r;
r.x = x; r.y = y; r.value = castle[x][y];
Q.push(r);
roomId[x][y] = roomNum;
rSize++;
while (!Q.empty()) {
Room r1, r2;
r1 = Q.front(); Q.pop();
for (int i = 0; i < 4; ++i) {
int dir = 1 << i;
if (0 == (dir&r1.value)) {
int tx = r1.x + dirX[i];
int ty = r1.y + dirY[i];
if (tx >= 1 && tx <= N && ty >= 1 && ty <= M && 0 == roomId[tx][ty]) {
roomId[tx][ty] = roomNum;
r2.x = tx;
r2.y = ty;
r2.value = castle[tx][ty];
rSize++;
Q.push(r2);
}
}
}
}
maxSize1 = max(maxSize1, rSize);
roomSize[roomNum] = rSize;
}
int main() {
#ifdef MARK
freopen("castle.in", "r", stdin);
freopen("castle.out", "w", stdout);
#endif // MARK
while (cin >> M >> N) {
for (int i = 1; i <= N; ++i) {
for (int j = 1; j <= M; ++j) {
cin >> castle[i][j];
roomId[i][j] = 0;
}
}
maxSize1 = 0;
int roomNum = 1;
roomSize.clear();
for (int i = 1; i <= N; ++i) {
for (int j = 1; j <= M; ++j) {
if (0 == roomId[i][j]) {
findRoom(i, j, roomNum);
roomNum++;
}
}
}
cout << roomNum-1 << endl << maxSize1 << endl;
// 枚举所有的墙壁,房间从左下往右上方向,墙壁优先选择N,其次为E
maxSize2 = 0;
int wallX, wallY;
char wallDir;
for (int y = 1; y <= M; ++y) {
for (int x = N; x >= 1; --x) {
int value = castle[x][y];
// N 方向
if ((value & 2) != 0) {
int tx = x + dirX[1];
int ty = y + dirY[1];
if (tx >= 1 && tx <= N && ty >= 1 && ty <= M && roomId[x][y] != roomId[tx][ty]) {
int totSize = roomSize[ roomId[x][y] ] + roomSize[ roomId[tx][ty] ];
if (totSize > maxSize2) {
maxSize2 = totSize;
wallX = x;
wallY = y;
wallDir = 'N';
}
}
}
// E 方向
if ((value & 4) != 0) {
int tx = x + dirX[2];
int ty = y + dirY[2];
if (tx >= 1 && tx <= N && ty >= 1 && ty <= M && roomId[x][y] != roomId[tx][ty]) {
int totSize = roomSize[ roomId[x][y] ] + roomSize[ roomId[tx][ty] ];
if (totSize > maxSize2) {
maxSize2 = totSize;
wallX = x;
wallY = y;
wallDir = 'E';
}
}
}
}
}
cout << maxSize2 << endl;
cout << wallX << " " << wallY << " " << wallDir << endl;
}
return 0;
}
USACO Section 2.1 The Castle 解题报告的更多相关文章
- USACO Section 1.3 Prime Cryptarithm 解题报告
题目 题目描述 牛式的定义,我们首先需要看下面这个算式结构: * * * x * * ------- * * * <-- partial product 1 * * * <-- parti ...
- USACO Section 1.4 Arithmetic Progressions 解题报告
题目 题目描述 现在给你一个数集,里面的数字都是由p^2+q^2这种形式构成的0 <= p,q <= M,我现在需要你在其中找出一个长为N的等差数列,数列中的第一个数字为a,公差为b,当你 ...
- USACO Section 1.3 Combination Lock 解题报告
题目 题目描述 农夫John的牛从农场逃脱出去了,所以他决定用一个密码锁来把农场的门锁起来,这个密码锁有三个表盘,每个表盘都是环形的,而且上面刻有1~N,现在John设了一个开锁密码,而且这个锁的设计 ...
- USACO Section 1.3 Barn Repair 解题报告
题目 题目描述 某农夫有一个养牛场,所有的牛圈都相邻的排成一排(共有S个牛圈),每个牛圈里面最多只圈养一头牛.有一天狂风卷积着乌云,电闪雷鸣,把牛圈的门给刮走了.幸运的是,有些牛因为放假,所以没在自己 ...
- USACO Section 1.3 Mixing Milk 解题报告
题目 题目描述 Merry Milk Makers 公司的业务是销售牛奶.它从农夫那里收购N单位的牛奶,然后销售出去.现在有M个农夫,每个农夫都存有一定量的牛奶,而且每个农夫都会有自己的定价.假设所有 ...
- USACO Section 1.2 Dual Palindromes 解题报告
题目 题目描述 有一些数(如 21),在十进制时不是回文数,但在其它进制(如二进制时为 10101)时就是回文数. 编一个程序,从文件读入两个十进制数N.S.然后找出前 N 个满足大于 S 且在两种以 ...
- USACO Section 1.2 Palindromic Squares 解题报告
题目 题目描述 输入一个基数B,现在要从1到300之间找出一些符合要求的数字N.如果N的平方转换成B进制数之后是一个回文串,那么N就符合要求.我们将N转换成B进制数输出,然后再将N的平方转换成B进制数 ...
- USACO Section 1.2 Milking Cows 解题报告
题目 题目描述 有3个农夫每天早上五点钟便起床去挤牛奶,现在第一个农夫挤牛奶的时刻为300(五点钟之后的第300个分钟开始),1000的时候结束.第二个农夫从700开始,1200结束.最后一个农夫从1 ...
- USACO Section 1.1 Broken Necklace 解题报告
题目 题目描述 有一串项链,它是由红蓝白三种颜色的珠子组成的,b代表蓝色,w代表白色,r代表红色,当它完整的时候是一个闭合的环形.现在它在某一个节点断裂了,之前的环形也随之变成了直线形.从两端开始收集 ...
随机推荐
- 【H5】滚动事件(jq)
$(function(){ console.log($('html,body').scrollTop()); //记录滚动高度(滚动前) }) $('html,body').scroll(funct ...
- VS编程,WPF单独更改TextBlock中部分文字格式的一种方法
原文:VS编程,WPF单独更改TextBlock中部分文字格式的一种方法 版权声明:我不生产代码,我只是代码的搬运工. https://blog.csdn.net/qq_43307934/articl ...
- 7、Class文件的格式
Class文件的格式 1.magic(魔数) 身份标识,用来标记这是不是一个CLASS文件 CLASS的魔数比较有浪漫气息,是0xCAFEBABE(咖啡宝贝),也标识着将来JAVA咖啡商标: 2.之后 ...
- [原创]STM32 BOOT模式配置以及作用
一.三种BOOT模式介绍 所谓启动,一般来说就是指我们下好程序后,重启芯片时,SYSCLK的第4个上升沿,BOOT引脚的值将被锁存.用户可以通过设置BOOT1和BOOT0引脚的状态,来选择在复位后的启 ...
- Yeoman的好基友:Grunt
grunt介绍 前端不能承受之痛 1.这是我们的生活 文件压缩:YUI Compressor.Google Closure 文件合并:fiddler + qzmin 文件校验:jshint 雪碧图:c ...
- Jq_浏览器兼容性及其浏览器版本
JQuery 中用 方法 jQuery.browser 来判断浏览器,返回值可以为: safari opera msie mozilla. 当然有时候我们还需要区分版本 这就要用到 jQuery.br ...
- Python对Selenium调用浏览器进行封装包括启用无头浏览器,及对应的浏览器配置文件
""" 获取浏览器 打开本地浏览器 打开远程浏览器 关闭浏览器 打开网址 最大化 最小化 标题 url 刷新 Python对Selenium封装浏览器调用 ------b ...
- 上云利器,K8S应用编排设计器之快到极致
前言在前面的文章中,我们已经提到,华为云有一个上云利器:应用编排设计器.作为华为云应用编排服务与用户沟通的桥梁,设计器坚持用户体验至上的理念,以图形化方式,在鼠标点击之间,助力企业快速上云.优质的交互 ...
- dokuwiki编辑器修改-color插件-添加按钮
需求 dokuwiki的编辑工具栏是以 MediaWiki 的为基础发展来的. 在它的编辑器color插件的颜色按钮中,我想添加新的按钮功能.如红色字体黄色背景的修饰,类似于涂中文字强调的意思. 步骤 ...
- ubuntu16.04下Hyperledger之搭建Fabric环境简单操作(五步启动e2e_cli)
如果你已经安装好go等工具.git及checkout相关代及下载相关镜像,您只需下面5步就能up e2e_cli~/go/src/github.com/hyperledger/fabric$ sudo ...