来自我的博客 王者编程大赛之二 — 蓄水池
自如寓打算门口用砖头围立一个蓄水池子,从上面看凹凸不平,凹的地方会有积水。那如果用数字代表每个砖头的高度,就形成一个二维数据(如示例),请问这个池子能存储多少单位的水?
例如二维数组为:
9 9 9 9
3 0 0 9
7 8 2 6
时,答案是中间的 0,0 位置可以存储 2(因为其外面最低是 2)个单位的水,因此答案为 2 + 2 = 4。
示例:
输入:[1 1 1 1,1 0 0 1,1 1 1 1]
输出:2
输入:[12 11 12 0 13,12 9 8 12 12,13 10 0 3 15,19 4 4 7 15,19 4 3 0 15,12 13 10 15 13]
输出:58
这道题是所有题中困惑我时间最长的题,一开始思维禁锢在想直接通过找到每块砖的四周有效最低砖高度 $H_{min}$,然后这块砖所剩的水为 $w\[i\]\[j\] = H_{min}-h\[i\]\[j\]$($h\[i\]\[j\]$ 为砖的高度,i 和 j 为砖的位置坐标),因此蓄水池能蓄下的水为 $\\sum_{i=1}^n\\sum_{j=1}^n w\[i\]\[j\]$。经过一番尝试,发现寻找某块砖四周最低有效砖逻辑比较复杂,且不易理解,又尝试过使用回溯算法寻找出池子中的所有连通图,但是也未有果。
最后,发现基础平台一位同学的实现思路很清晰,我认为他的实现是最合适的,所以研究了一下。该实现中机智地采用逆向思维,[首先往池子注满水(最高砖的高度),然后再通过条件判定每块砖是否需要进行漏水,一直到没有砖需要进行漏水操作]()。
实现思路如下:
算法流程图示如下:
[
](https://img.fanhaobai.com/2017/12/2017-ziroom-king-2/53afcc7a-ff51-4191-aab0-5c9cc6850566.png)
实现的类结构如下,特殊的方法已提取出,并将一一详细说明。
<?php
class Pool
{
public $gridArray = array();
public $maxHeight = 0;
public $row = 0;
public $col = 0;
public function __construct(array $data)
{
$this->row = count($data);
$this->col = count($data[0]);
foreach ($data as $row => $rowArray) {
foreach ($rowArray as $col => $height) {
$height = (int)$height;
$this->gridArray[$row][$col]['height'] = $height;
$this->gridArray[$row][$col]['water'] = 0;
//获取最高砖的高度
if ($this->maxHeight < $height) {
$this->maxHeight = $height;
}
}
}
}
//判断是否是水池边界
public function isBorder($row, $col)
{
if ($row == 0
|| $row == $this->row - 1
|| $col == 0
|| $col == $this->col - 1
) {
return true;
}
return false;
}
public function run()
{
$this->addWater();
while ($this->removeWater()) ;
return $this->collect();
}
}
注水操作:
public function addWater()
{
foreach ($this->gridArray as $row => $rowArray) {
foreach ($rowArray as $col => $grid) {
if (!$this->isBorder($row, $col)) {
$this->gridArray[$row][$col]['water'] = $this->maxHeight - $this->gridArray[$row][$col]['height'];
}
}
}
}
漏水操作:
public function removeWater()
{
foreach ($this->gridArray as $row => $rowArray) {
foreach ($rowArray as $col => $grid) {
if ($this->canRemove($row, $col)) {
return true;
}
}
}
return false;
}
漏水条件实现如下:
public function canRemove($row, $col)
{
$can = false;
if ($this->gridArray[$row][$col]['water'] > 0) {
//上
if ($this->gridArray[$row][$col]['water'] + $this->gridArray[$row][$col]['height'] >
$this->gridArray[$row - 1][$col]['water'] + $this->gridArray[$row - 1][$col]['height']) {
$this->gridArray[$row][$col]['water'] =
$this->gridArray[$row - 1][$col]['water'] + $this->gridArray[$row - 1][$col]['height']
- $this->gridArray[$row][$col]['height'];
if ($this->gridArray[$row][$col]['water'] < 0) {
$this->gridArray[$row][$col]['water'] = 0;
}
$can = true;
}
//右
if ($this->gridArray[$row][$col]['water'] + $this->gridArray[$row][$col]['height'] >
$this->gridArray[$row][$col + 1]['water'] + $this->gridArray[$row][$col + 1]['height']) {
$this->gridArray[$row][$col]['water'] =
$this->gridArray[$row][$col + 1]['water'] + $this->gridArray[$row][$col + 1]['height']
- $this->gridArray[$row][$col]['height'];
if ($this->gridArray[$row][$col]['water'] < 0) {
$this->gridArray[$row][$col]['water'] = 0;
}
$can = true;
}
//下
if ($this->gridArray[$row][$col]['water'] + $this->gridArray[$row][$col]['height'] >
$this->gridArray[$row + 1][$col]['water'] + $this->gridArray[$row + 1][$col]['height']) {
$this->gridArray[$row][$col]['water'] =
$this->gridArray[$row + 1][$col]['water'] + $this->gridArray[$row + 1][$col]['height']
- $this->gridArray[$row][$col]['height'];
if ($this->gridArray[$row][$col]['water'] < 0) {
$this->gridArray[$row][$col]['water'] = 0;
}
$can = true;
}
//左
if ($this->gridArray[$row][$col]['water'] + $this->gridArray[$row][$col]['height'] >
$this->gridArray[$row][$col - 1]['water'] + $this->gridArray[$row][$col - 1]['height']) {
$this->gridArray[$row][$col]['water'] =
$this->gridArray[$row][$col - 1]['water'] + $this->gridArray[$row][$col - 1]['height']
- $this->gridArray[$row][$col]['height'];
if ($this->gridArray[$row][$col]['water'] < 0) {
$this->gridArray[$row][$col]['water'] = 0;
}
$can = true;
}
}
return $can;
}
持续漏水操作:
public function run()
{
while ($this->removeWater()) ;
}
求和砖的盛水量:
public function collect()
{
$sum = 0;
foreach ($this->gridArray as $row => $rowArray) {
foreach ($rowArray as $col => $grid) {
$sum += $grid['water'];
}
}
return $sum;
}
接收标准输入处理并输出结果:
$filter = function ($value) {
return explode(' ', $value);
};
$pool = new Pool(array_map($filter, explode(',', $input)));
echo $pool->run(), PHP_EOL;
Twitter 之前曾经出过类似蓄水池的笔试题,只不过本题是立体水池(二维数组),Twitter 蓄水池笔试题是平面水池(一维数组),解题复杂度也就降低了,当然 Twitter 蓄水池笔试题也可以采用本题的思想来实现,但是时间复杂度为 $O(n^2)$,采用 我的Twitter技术面试失败了 的实现时间复杂度为 $O(n)$。
[](https://img.fanhaobai.com/2017/12/2017-ziroom-king-2/b2f137ff-b91b-4036-b7d8-fc86f82fb980.png)
[
](https://img.fanhaobai.com/2017/12/2017-ziroom-king-2/b2f137ff-b91b-4036-b7d8-fc86f82fb980.png)
实现思路如下:
具体实现,请直接参考 CuGBabyBeaR 文章。
本题的蓄水池问题,如果理解了问题本质并逆向思维,将寻找某块砖四周最低有效砖高度(寻找有效砖涉及到边界扩散)转化为判断某块砖是否需要漏水条件,那么问题就简化很多了,那后续编码也就很容易实现了,本文算法的时间复杂度为 $O(n^3)$。
相关文章 [»]()
Original url: Access
Created at: 2018-10-10 15:33:15
Category: default
Tags: none
未标明原创文章均为采集,版权归作者所有,转载无需和我联系,请注明原出处,南摩阿彌陀佛,知识,不只知道,要得到
java windows火焰图_mob64ca12ec8020的技术博客_51CTO博客 - 在windows下不可行,不知道作者是怎样搞的 监听SpringBoot 服务启动成功事件并打印信息_监听springboot启动完毕-CSDN博客 SpringBoot中就绪探针和存活探针_management.endpoint.health.probes.enabled-CSDN博客 u2u转换板 - 嘉立创EDA开源硬件平台 Spring Boot 项目的轻量级 HTTP 客户端 retrofit 框架,快来试试它!_Java精选-CSDN博客 手把手教你打造一套最牛的知识笔记管理系统! - 知乎 - 想法有重合-理论可参考 安宇雨 闲鱼 机械键盘 客制化 开贴记录 文本 linux 使用find命令查找包含某字符串的文件_beijihukk的博客-CSDN博客_find 查找字符串 ---- mac 也适用 安宇雨 打字音 记录集合 B站 bilibili 自行搭建 开坑 真正的客制化 安宇雨 黑苹果开坑 查找工具包maven pom 引用地 工具网站 Dantelis 介绍的玩轴入坑攻略 --- 关于轴的一些说法 --- 非官方 ---- 心得而已 --- 长期开坑更新 [本人问题][新开坑位]关于自动化测试的工具与平台应用 机械键盘 开团 网站记录 -- 能做一个收集的程序就好了 不过现在没时间 -- 信息大多是在群里发的 - 你要让垃圾佬 都去一个地方看难度也是很大的 精神支柱 [超级前台]sprinbboot maven superdesk-app 记录 [信息有用] [环境准备] [基本完成] [sebp/elk] 给已创建的Docker容器增加新的端口映射 - qq_30599553的博客 - CSDN博客 [正在研究] Elasticsearch, Logstash, Kibana (ELK) Docker image documentation elasticsearch centos 安装记录 及 启动手记 正式服务器 39 elasticsearch 问题合集 不断更新 6.1.1 | 6.5.1 两个版本 博客程序 - 测试 - bug记录 等等问题 laravel的启动过程解析 - lpfuture - 博客园 OAuth2 Server PHP 用 Laravel 搭建带 OAuth2 验证的 RESTful 服务 | Laravel China 社区 - 高品质的 Laravel 和 PHP 开发者社区 利用Laravel 搭建oauth2 API接口 附 Unauthenticated 解决办法 - 煮茶的博客 - SegmentFault 思否 使用 OAuth2-Server-php 搭建 OAuth2 Server - 午时的海 - 博客园 基于PHP构建OAuth 2.0 服务端 认证平台 - Endv - 博客园 Laravel 的 Artisan 命令行工具 Laravel 的文件系统和云存储功能集成 浅谈Chromium中的设计模式--终--Observer模式 浅谈Chromium中的设计模式--二--pre/post和Delegate模式 浅谈Chromium中的设计模式--一--Chromium中模块分层和进程模型 DeepMind 4 Hacking Yourself README.md update 20211011
Laravel China 简书 知乎 博客园 CSDN博客 开源中国 Go Further Ryan是菜鸟 | LNMP技术栈笔记 云栖社区-阿里云 Netflix技术博客 Techie Delight Linkedin技术博客 Dropbox技术博客 Facebook技术博客 淘宝中间件团队 美团技术博客 360技术博客 古巷博客 - 一个专注于分享的不正常博客 软件测试知识传播 - 测试窝 有赞技术团队 阮一峰 语雀 静觅丨崔庆才的个人博客 软件测试从业者综合能力提升 - isTester IBM Java 开发 使用开放 Java 生态系统开发现代应用程序 pengdai 一个强大的博主 HTML5资源教程 | 分享HTML5开发资源和开发教程 蘑菇博客 - 专注于技术分享的博客平台 个人博客-leapMie 流星007 CSDN博客 - 舍其小伙伴 稀土掘金 Go 技术论坛 | Golang / Go 语言中国知识社区
最新评论