C语言编程题_3D接雨水

接雨水的题目描述如下。

(1) 2D接雨水: 字节员工是不是个个都会接雨水 ;

(2) 3D接雨水: 407. 接雨水 II ;

(3) 3D接雨水: 字节人都会的 3D接雨水 。

  问题描述

  难度:困难

  给你一个 m x n 的矩阵,其中的值均为非负整数,代表二维高度图每个单元的高度,请计算图中形状最多能接多少体积的雨水。

  示例1:

  输入:heightMap = [[1,4,3,1,3,2],[3,2,1,3,2,4],[2,3,3,2,3,1]];

  输出:4

  解释:下雨后,雨水将会上图蓝色的方块中。总的接雨水量为1+1+1=4。

  我找到的最好的思路是:把这个看成木桶接水,就像是短板理论。最外层一圈一定接不到水,也就是可以看成木桶的边缘,先把木桶的边缘都放进优先队列中,这时取出最小的柱子,向这个柱子的 4 个方向扩散,扩散出的柱子有两种情况,如果比当前柱子大,那么不能接水,如果比当前柱子小,那么接到的水就是当前柱子高度减去扩散柱子的高度(为什么?因为当前柱子一定是边缘最小的了,所以它一定能接到水),后面在把扩散的柱子加入到优先队列中,已经比较完的柱子就移出队列,这样就又形成了新的桶的边缘,并且水桶面积也在不断缩小(当然要记录走过的柱子,防止重复走),最终就会计算完成。

  下面是用C语言实现的3D接雨水:

#include <stdio.h>
#include <stdlib.h>

#define Cols            (6)
#define Rows            (3)

#define Status_ToDo     (0) // 待处理的。
#define Status_Done     (1) // 已处理的。
#define Status_Border   (2) // 作为木桶板。

int addRainWaterOfTBLR(int height[][Cols], int status[][Cols], int minBorderHeight, int row, int col);

void initStatus(int arr[][Cols]) {
    // // 注: 使用下面语句得到的rowCount为0, 因为arr作为形参的时候就会被当作一个指针。
    // int rowCount = sizeof(arr) / sizeof(arr[0]);
    for (int row = 0; row < Rows; row++) {
        for (int col = 0; col < Cols; col++) {
            if (0 == row || Rows-1 == row || 0 == col || Cols-1 == col) {
                arr[row][col] = Status_Border;
            } else {
                arr[row][col] = Status_ToDo;
            }
        }
    }
    // 去掉四个角。
    arr[0][0] = arr[0][Cols-1] = arr[Rows-1][0] = arr[Rows-1][Cols-1] = Status_Done;
}

void showStatus(int height[][Cols], int status[][Cols]) {
    printf("打印状态:\n");
    for (int row = 0; row < Rows; row++) {
        for (int col = 0; col < Cols; col++) {
            printf("%d(%d), ", height[row][col], status[row][col]);
        }
        printf("\n");
    }
}

void findPositionOfMinBorder(int height[][Cols], int status[][Cols], int result[2]) {
    // printf("findPositionOfMinBorder\n");
    result[0] = 0;
    result[1] = 0;
    for (int row = 0; row < Rows; row++) {
        for (int col = 0; col < Cols; col++) {
            if (Status_Border == status[row][col]
                && ((0 == result[0] && 0 == result[1]) || height[row][col] < height[result[0]][result[1]])
            ) {
                result[0] = row;
                result[1] = col;
            }
        }
    }
    printf("最小板的位置是(%d,%d)\n", result[0], result[1]);
}

void repairBorder(int status[][Cols]) {
    for (int row = 0; row < Rows; row++) {
        for (int col = 0; col < Cols; col++) {
            if (Status_Border == status[row][col]) {
                //上下左右都不是待处理的。
                if ((0 == row || (row > 0 && status[row-1][col] != Status_ToDo))
                    && (Rows-1 == row || (Rows-1 > row && status[row+1][col] != Status_ToDo))
                    && (0 == col || (col > 0 && status[row][col-1] != Status_ToDo))
                    && (Cols-1 == col || (Cols-1 > col && status[row][col+1] != Status_ToDo))
                ) {
                    status[row][col] = Status_Done;
                }
            }
        }
    }
}

int addRainWaterAtPosition(int height[][Cols], int status[][Cols], int minBorderHeight, int row, int col) {
    int result = 0;
    // printf("addRainWaterAtPosition(%d,%d)\n", row, col);
    if (height[row][col] <= minBorderHeight) {
        result += (minBorderHeight - height[row][col]);
        result += addRainWaterOfTBLR(height, status, minBorderHeight, row, col);
    } else {
        status[row][col] = Status_Border;
    }
    return result;
}

int addRainWaterOfTBLR(int height[][Cols], int status[][Cols], int minBorderHeight, int row, int col) {
    int result = 0;
    // printf("addRainWaterOfTBLR(%d,%d)\n", row, col);
    status[row][col] = Status_Done;
    if (row > 0 && Status_ToDo == status[row-1][col]) { // 上。
        result += addRainWaterAtPosition(height, status, minBorderHeight, row-1, col);
    }
    if (row < Rows-1 && Status_ToDo == status[row+1][col]) { // 下。
        result += addRainWaterAtPosition(height, status, minBorderHeight, row+1, col);
    }
    if (col > 0 && Status_ToDo == status[row][col-1]) { // 左。
        result += addRainWaterAtPosition(height, status, minBorderHeight, row, col-1);
    }
    if (col < Cols-1 && Status_ToDo == status[row][col+1]) { // 右。
        result += addRainWaterAtPosition(height, status, minBorderHeight, row, col+1);
    }
    return result;
}

// 判断完成。
int checkFinish(int status[][Cols]) {
    for (int row = 0; row < Rows; row++) {
        for (int col = 0; col < Cols; col++) {
            if (Status_ToDo == status[row][col]) {
                return 0;
            }
        }
    }
    return 1;
}

// 简介:  3D接雨水。
int main(void)
{
    int rainWater = 0;
    int heightMap[][Cols] = {{1,4,3,1,3,2},{3,2,1,3,2,4},{2,3,3,2,3,1}};
    // printf("heightMap_rowCount = %d。\n", sizeof(heightMap) / sizeof(heightMap[0]));
    int (*status)[Cols] = malloc(sizeof(int) * (sizeof(heightMap) / sizeof(heightMap[0])) * Cols);
    initStatus(status);
    showStatus(heightMap, status);

    while(!checkFinish(status)) {
        int positionOfMinBorder[2] = {0, 0};
        findPositionOfMinBorder(heightMap, status, positionOfMinBorder);
        int minBorderHeight = heightMap[positionOfMinBorder[0]][positionOfMinBorder[1]];
        // printf("minBorderHeight=%d\n", minBorderHeight);
        rainWater += addRainWaterOfTBLR(heightMap, status, minBorderHeight, positionOfMinBorder[0], positionOfMinBorder[1]);
        repairBorder(status);
        showStatus(heightMap, status);
    }
    printf("总的接雨水量为%d\n", rainWater);

    free(status);

    printf("程序正常退出。\n");
    return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

企业有哪些常见网络需求场景?

企业的网络场景需求多种多样&#xff0c;主要取决于其业务规模、运营模式、技术应用等因素。 常见的企业网络场景需求 办公网络需求&#xff1a; 高速稳定的内部网络连接&#xff0c;以支持员工日常办公、数据传输和资源共享。 无线办公网络覆盖&#xff0c;以便员工在会议室…

OpenCV从入门到精通实战(七)——探索图像处理:自定义滤波与OpenCV卷积核

本文主要介绍如何使用Python和OpenCV库通过卷积操作来应用不同的图像滤波效果。主要分为几个步骤&#xff1a;图像的读取与处理、自定义卷积函数的实现、不同卷积核的应用&#xff0c;以及结果的展示。 卷积 在图像处理中&#xff0c;卷积是一种重要的操作&#xff0c;它通过…

C++|运算符重载(3)|日期类的计算

前面介绍了运算符重载相关规则和方法&#xff0c;今天用运算重载函数实现对日期类的操作。 目录 前面准备 实现功能&#xff1a; -运算符 Date类和int 相减 Date类和Date类相减 运算符 &#xff0c;-运算符 ,!运算符 >,>运算符 <,<运算符 &#xff0c;-…

前端vue仿美团风格下拉筛选框在前端开发中的实现与应用

摘要&#xff1a; 在前端开发中&#xff0c;下拉筛选框是提升用户体验和交互效果的重要组件之一。本文将以美团风格的下拉筛选框为例&#xff0c;介绍其实现原理、技术细节以及在实际项目中的应用。通过自定义组件CCDropDownFilter&#xff0c;我们将展示如何创建一个功能丰富、…

蚓链数字化营销系统与数字资产的关系

蚓链数字化营销系统是一种利用数字技术来实现营销目标的系统。它集成了多种数字营销工具和渠道&#xff0c;以收集、分析和利用客户数据&#xff0c;优化营销活动&#xff0c;并提高营销效果。 数字资产是一种新型的资产类别&#xff0c;它们以电子数据的形式存在&#xff0c;可…

快速了解-BTP

名词了解 BTP&#xff1a;SAP Business Technology Platform 是一个技术和业务的平台ETWEAVER &#xff08;SAP NW&#xff09;&#xff1a;NetWeaver本质上是SAP一系列技术产品的集成平台PAAS Cloud Foundry&#xff08;云原生&#xff09;&#xff1a;开源云服务平台烟囱式…

解决Android studio更换sdk地址后flutter项目显示no device selected

问题描述 因为之前sdk的路径在c盘上,经常在更新或下在sdk后c盘饱满,于是就更换了sdk的路径,更换sdk路径后就导致flutter项目在选择设备的时候出现no device selected 找不到设备,但是在device Manager可以看到物理设备或者是虚拟设备。如下图所示。 问题分析 导致这个问题…

【网安小白成长之路】9.sql注入操作

&#x1f42e;博主syst1m 带你 acquire knowledge&#xff01; ✨博客首页——syst1m的博客&#x1f498; &#x1f51e; 《网安小白成长之路(我要变成大佬&#x1f60e;&#xff01;&#xff01;)》真实小白学习历程&#xff0c;手把手带你一起从入门到入狱&#x1f6ad; &…

基于springboot实现实验室管理系统项目【项目源码+论文说明】

基于springboot实现实验室管理系统演示 摘要 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在技术上已逐步成熟。本文介绍了实验室管理系统的开发全过程。通过分析实验室管理系统管理的不足&#xff0c;创建了一个计算机管理实验室管理系统的方案…

使用Hypothesis生成测试数据

Hypothesis是Python的一个高级测试库。它允许编写测试用例时参数化&#xff0c;然后生成使测试失败的简单易懂的测试数据。可以用更少的工作在代码中发现更多的bug。 安装 pip install hypothesis如何设计测试数据 通过介绍也许你还不了解它是干嘛的&#xff0c;没关系&…

Redis面试题三(集群)

目录 1.Redis 集群搭建有几种模式 2.Redis 主从复制的实现 全量同步 增量同步 3.Redis 的主从同步策略 1. 全量同步&#xff08;Full Resynchronization&#xff09; 2. 增量同步&#xff08;Incremental Replication&#xff09; 4.Redis一致性hash 基本原理 节点动态…

使用shared lib将各个构建工具集成到一起

共享库代码 package devopsdef Build(buildType, buildShell){def buildTools ["mvn": "MVN", "ant": "ANT", "gradle": "GRADLE"]println("当前buildType是${buildType}")buildHome tool buildTool…

记内网http洪水攻击,导致网页无法访问一事

事由 最近两日&#xff0c;部分同事在访问税纪云平台时&#xff0c;登录跳转页面频繁转圈、要么就是出现无法连接的错误提示。 无法访问此页面 已重置连接。 请尝试: 检查连接 检查代理和防火墙 运行 Windows 网络诊断经过以下几方面的排查&#xff0c;无果。 后续通过检查…

【C++】从零开始认识泛型编程 — 模版

送给大家一句话&#xff1a; 尽管眼下十分艰难&#xff0c;可日后这段经历说不定就会开花结果。总有一天我们都会成为别人的回忆&#xff0c;所以尽力让它美好吧。 – 岩井俊二 &#xff3c;&#xff3c;\ ⱶ˝୧(๑ ⁼̴̀ᐜ⁼̴́๑)૭兯 //&#xff0f;&#xff0f; &#…

pycharm安装AI写代码插件

在IDE安装特慢&#xff08;可能找不到插件&#xff09; 去官网搜一下 对应安装包 下载zip在IDE解压 插件--已安装齿轮图标--从磁盘里安装 选择下载的插件 应用 --重启OK

3gp转MP4怎么转?简单的3个方法~

3gp最初是由第三代合作伙伴计划&#xff08;3rd Generation Partnership Project&#xff09;设计的&#xff0c;旨在满足移动设备对高效传输音频和视频的需求。它的起源可以追溯到20世纪初&#xff0c;当时移动通信技术的飞速发展使得人们对更加高效的多媒体文件格式有了迫切需…

几种免费SSL证书申请方式

目录 DV单域名免费证书的获取渠道&#xff1a; DV多域名免费证书获取渠道&#xff1a; DV通配符免费证书获取渠道&#xff1a; 随着现在网络安全意识的逐渐提升&#xff0c;越来越多的网站都在相继配对部署SSL证书&#xff0c;用以实现https访问。 大家都知道SSL证书好&…

【linux】基础IO(软硬链接)

上一节我们已经搞懂了已经被打开的文件&#xff0c;还有没有被打开的文件都是怎样被管理起来的&#xff0c;同样&#xff0c;路径的重要性也不言而喻&#xff0c;是确定文件在那个分区&#xff0c;进而可以解析到目标文件与目录内容的关系&#xff0c;从而找到inode&#xff0c…

微带线设计细节的模拟仿真分析

微带线设计在很多PCB设计场景中被应用&#xff0c;但是&#xff0c;有些工程师往往并不注重设计细节&#xff0c;导致最后的设计指标与预期相差甚远&#xff0c;比如设计中&#xff0c;会将丝印、散热金属等放在走线的上方&#xff0c;设计检查时会有人产生质疑&#xff0c;至于…

3DTiles生产流程与规范

一篇19年整理的比较老的笔记了。更多精彩内容尽在数字孪生平台。 瓦片切分 标准的四叉树切分对于均匀分布的地理数据切片非常有效&#xff0c;但是这样均等的切分不适用于随机分布、不均匀分布的地理数据&#xff0c;当地理数据稀疏分布的时候&#xff0c;均等的四叉树就不再高…
最新文章