[蓝桥杯] 挖矿(CC++双语版)

题目链接

          P10904 [蓝桥杯 2024 省 C] 挖矿 - 洛谷

题目理解

        我们可以将这道题中矿洞的位置理解成为一个坐标轴,以题目样例绘出坐标轴:

样例:

        输入的5为矿洞数量,4为可走的步数。第二行输入是5个矿洞的坐标。输出结果为在要求步数内最多可采集到的单位数量的矿石。

         我们下面来对样例进行分析:

        初始状态如下:

        初始位置为0,剩余步数为4。当坐标0有矿石时,不需要消耗步数,可直接获取矿石:

        我们向-1移动:

        此时若是向负轴移动,我们只能采到1个单位的矿石,正轴方向可以采到2个单位。于是我们向正轴方向进发:

        继续进发:

        继续:

 

解题思路

      ——(篇幅较长,大家也可以先看详细注释后的代码,遇到不懂的地方再看解题思路)——

   一、核心思路概述

        解决该挖矿问题的核心在于合理利用数轴上矿洞的分布信息以及移动距离限制,通过巧妙的统计和计算,找出在给定移动距离内能够挖掘到最多矿石的方法。主要步骤包括对矿洞坐标的分类统计、构建前缀和数组以快速计算区间矿洞数量,以及全面考虑不同移动策略下的矿石获取情况。

   二、具体步骤解析

    (一)输入数据处理

1. 读取矿洞数量 n 和最大移动距离 m,这两个参数将决定后续计算的范围和约束条件。

2. 依次读取 n 个矿洞在数轴上的坐标值,对于每个坐标值进行如下分类处理:

  •         如果坐标值为 0,说明该矿洞位于小蓝的起始位置,直接将此类矿洞的计数变量 s 加 1。这些矿洞无需移动即可挖掘,对最终结果有直接贡献。
  •         如果坐标值不为 0,且其绝对值小于等于 m,则根据坐标值的正负性分别处理。
  •         若坐标值为负数,将其对应在负半轴的位置索引处的计数器(例如数组 l)加 1,用于统计负半轴上在移动距离范围内的矿洞数量。
  •         若坐标值为正数,将其对应在正半轴的位置索引处的计数器(例如数组 r)加 1,用于统计正半轴上在移动距离范围内的矿洞数量。

  (二)构建前缀和数组

        1. 对于记录负半轴矿洞数量的数组 l,从索引 1 开始到 m,依次计算前缀和。即 l[i] = l[i - 1] + l[i],这样 l[i] 就表示数轴负半轴上从 -1 到 -i 位置的矿洞总数。通过前缀和,我们可以在后续计算中快速得到负半轴某一区间内的矿洞数量。

        2. 同理,对于记录正半轴矿洞数量的数组 r,也从索引 1 开始到 m,计算前缀和 r[i] = r[i - 1] + r[i],使得 r[i] 表示数轴正半轴上从 1 到 i 位置的矿洞总数。

  (三)计算最大矿石获取量

1. 初始假设:

  •         首先假设最大矿石获取量 ans 为只在正半轴或只在负半轴移动时能够挖掘到的最大矿石数。即 ans = max(r[m], l[m]),这是一种简单的初步情况考虑,因为在某些情况下,仅在一个方向移动可能就能够获取最多的矿石。

2. 考虑混合移动策略:

  •         通过循环遍历 i 从 1 到 m / 2(因为左右移动距离之和不能超过 m,所以 i 最大取到 m / 2),尝试不同的左右移动组合。 - 对于每一个 i 值,计算两种混合移动策略下能够挖掘到的矿石数:
  •         先向右移动 i 单位距离,再向左移动 m - 2 * i 单位距离时,能够挖掘到的矿石数为 sr = r[i] + l[m - 2 * i]。这里 r[i] 表示向右移动 i 单位距离过程中挖掘到的正半轴矿洞数,l[m - 2 * i] 表示向左移动 m - 2 * i 单位距离过程中挖掘到的负半轴矿洞数。 - 先向左移动 i 单位距离,再向右移动 m - 2 * i 单位距离时,能够挖掘到的矿石数为 sl = l[i] + r[m - 2 * i]。这里 l[i] 表示向左移动 i 单位距离过程中挖掘到的负半轴矿洞数,r[m - 2 * i] 表示向右移动 m - 2 * i 单位距离过程中挖掘到的正半轴矿洞数。
  •         每次计算出 sr 和 sl 后,更新最大矿石获取量 ans,即 ans = max(ans, sr, sl),取当前的 ans、sr 和 sl 中的最大值作为新的 ans。

3. 加上起始点矿洞数量:

  •         最后,将起始点(坐标为 0)的矿洞数量 s 加到 ans 中,因为这些矿洞在初始位置就可获取,无需移动。此时得到的 ans 即为在移动距离不超过 m 的前提下,小蓝最多能获得的矿石单位数量。 通过以上步骤,我们可以全面且高效地解决在给定条件下的挖矿问题,找到获取最多矿石的方案。

完整代码(详细注释)

        本文由于用到了std库中的一些函数,所以作者最开始采用了C++,C语言代码作者用ai转化并确保代码无误后给大家放在了下面。

    1.C++代码

#include <bits/stdc++.h>// 定义 solve 函数来解决挖矿问题
void solve() 
{int n, m;// 从标准输入读取矿洞数量 n 和最大移动距离 mstd::cin >> n >> m;// s 用于记录坐标为 0 的矿洞数量int s = 0;// 定义两个静态数组 l 和 r 来分别记录负半轴和正半轴矿洞的前缀和// 数组大小为 m + 1,初始化为 0int l[10000001] = {0};int r[10000001] = {0};// 遍历每个矿洞for (int i = 0; i < n; ++i) {int x;// 读取当前矿洞的坐标std::cin >> x;// 如果矿洞坐标的绝对值小于等于最大移动距离 m 且为负数//abs为绝对值函数if (std::abs(x) <= m && x < 0){// 对应负半轴的位置计数加 1l[-x]++;}// 如果矿洞坐标的绝对值小于等于最大移动距离 m 且为正数else if (std::abs(x) <= m && x > 0) {// 对应正半轴的位置计数加 1r[x]++;}// 如果矿洞坐标为 0else if (x == 0) {// 坐标为 0 的矿洞数量加 1s++;}}// 计算负半轴矿洞的前缀和for (int i = 1; i <= m; ++i){l[i] += l[i - 1];}// 计算正半轴矿洞的前缀和for (int i = 1; i <= m; ++i) {r[i] += r[i - 1];}// 先假设最大矿石数为只走正半轴或只走负半轴能挖到的最大矿石数int ans = std::max(r[m], l[m]);// 尝试不同的左右移动组合,i 表示先向一个方向移动的距离for (int i = 1; i <= m / 2; ++i) {// 先向右移动 i 的距离,再向左移动 m - i * 2 的距离能挖到的矿石数int sr = r[i] + l[m - i * 2];// 先向左移动 i 的距离,再向右移动 m - i * 2 的距离能挖到的矿石数int sl = l[i] + r[m - i * 2];// 更新最大矿石数ans = std::max({ans, sr, sl});}// 加上坐标为 0 的矿洞数量ans += s;// 输出最大能获得的矿石数std::cout << ans << '\n';
}int main() 
{solve();return 0;
}

     2.C语言代码 

#include <stdio.h>
#include <math.h>#define MAX_M 10000001// 定义 solve 函数来解决挖矿问题
void solve() {int n, m;// 从标准输入读取矿洞数量 n 和最大移动距离 mscanf("%d %d", &n, &m);// s 用于记录坐标为 0 的矿洞数量int s = 0;// 定义两个静态数组 l 和 r 来分别记录负半轴和正半轴矿洞的前缀和// 数组大小为 MAX_M,初始化为 0int l[MAX_M] = {0};int r[MAX_M] = {0};// 遍历每个矿洞for (int i = 0; i < n; ++i) {int x;// 读取当前矿洞的坐标scanf("%d", &x);// 如果矿洞坐标的绝对值小于等于最大移动距离 m 且为负数if (abs(x) <= m && x < 0) {// 对应负半轴的位置计数加 1l[-x]++;}// 如果矿洞坐标的绝对值小于等于最大移动距离 m 且为正数else if (abs(x) <= m && x > 0) {// 对应正半轴的位置计数加 1r[x]++;}// 如果矿洞坐标为 0else if (x == 0) {// 坐标为 0 的矿洞数量加 1s++;}}// 计算负半轴矿洞的前缀和for (int i = 1; i <= m; ++i) {l[i] += l[i - 1];}// 计算正半轴矿洞的前缀和for (int i = 1; i <= m; ++i) {r[i] += r[i - 1];}// 先假设最大矿石数为只走正半轴或只走负半轴能挖到的最大矿石数int ans = (r[m] > l[m]) ? r[m] : l[m];// 尝试不同的左右移动组合,i 表示先向一个方向移动的距离for (int i = 1; i <= m / 2; ++i) {// 先向右移动 i 的距离,再向左移动 m - i * 2 的距离能挖到的矿石数int sr = r[i] + l[m - i * 2];// 先向左移动 i 的距离,再向右移动 m - i * 2 的距离能挖到的矿石数int sl = l[i] + r[m - i * 2];// 更新最大矿石数if (sr > ans) ans = sr;if (sl > ans) ans = sl;}// 加上坐标为 0 的矿洞数量ans += s;// 输出最大能获得的矿石数printf("%d\n", ans);
}int main() {solve();return 0;
}

       AC拿下!

疑难解答 

    1.max函数的运用

        用于比较出几个数中最大的数字,大家可以简单理解为两个数比较就不需要大括号,如果三个及以上的话需要大括号。

  • 两个参数比较:当比较两个值时,使用 std::max(a, b) 形式,这里 a 和 b 是要比较的对象,不需要大括号。比如 int num1 = 5, num2 = 3; int maxNum = std::max(num1, num2); 。
  • 三个及以上参数比较:对于三个或更多值,常使用接收初始化列表形式的重载,即 std::max({val1, val2, val3,...}) ,需要用大括号把这些值组成初始化列表传递给函数。像 int num1 = 1, num2 = 2, num3 = 3; int maxNum = std::max({num1, num2, num3}); 。

———(如有问题,欢迎评论区提问)———

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

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

相关文章

2025年Python的主要应用场景

李升伟 编译 Python在2025年仍是最受欢迎和强大的编程语言之一。其简洁易读的语法以及庞大的库生态系统&#xff0c;使其成为各行业开发者的首选。无论是构建复杂的数据管道&#xff0c;还是自动化重复性任务&#xff0c;Python都能提供广泛的应用场景&#xff0c;以实现快速、…

fastapi完全离线环境(无外网)的访问Swagger所做特殊处理

在互联网环境中&#xff0c;只要 启动FastAPI 服务运行在本地机器上&#xff0c;访问 http://localhost:8000/docs&#xff08;Swagger UI&#xff09;就可以访问到Swagger界面&#xff0c;但是在完全离线环境&#xff08;无外网&#xff09;下如何访问Swagger页面呢&#xff1…

Ubuntu 20.04 出现问号图标且无法联网 修复

在 Ubuntu 中遇到网络连接问题&#xff08;如出现问号图标且无法联网&#xff09;&#xff0c;可以通过以下命令尝试重启网络服务&#xff1a; 1. 推荐先修改DNS 编辑 -> 虚拟机网络编辑器-> VMnet8 ->NAT 设置 -> DNS 设置 -> 设置DNS 服务器 DNS填什么 取决…

哈希表(开散列)的实现

目录 引入 开散列的底层实现 哈希表的定义 哈希表的扩容 哈希表的插入 哈希表查找 哈希表的删除 引入 接上一篇&#xff0c;我们使用了闭散列的方法解决了哈希冲突&#xff0c;此篇文章将会使用开散列的方式解决哈希冲突&#xff0c;后面对unordered_set和unordered_map的…

机器学习(八):K-Means聚类原理与实战

声明&#xff1a;未经允许禁止转载与抄袭。 前言 k k k均值&#xff08; k k k-means&#xff09;聚类算法是一种经典的无监督聚类算法&#xff0c;本文将深入解析其理论原理&#xff0c;并在真是数据集上进行算法实践&#xff0c;话不多说&#xff0c;请看下文。 算法原理 …

判断矩阵A和矩阵B是否相似?

【例题1】 &#xff08;1&#xff09;方法1 &#xff08;2&#xff09;方法2 &#xff08;3&#xff09;方法3 好题\(^o^)/~ 【注意】当二次多项式有重根时&#xff0c;即判别式为零&#xff0c;此时二次多项式是完全平方。

【10】搭建k8s集群系列(二进制部署)之安装Dashboard和CoreDNS

一、部署Dashboard 1.1、创建kubernetes-dashboard.yaml文件 完整的yaml配置文件信息如下&#xff1a; # Copyright 2017 The Kubernetes Authors. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in …

大数据技术与Scala

集合高级函数 过滤 通过条件筛选集合元素&#xff0c;返回新集合。 映射 对每个元素应用函数&#xff0c;生成新集集合 扁平化 将嵌套集合展平为单层集合。 扁平化映射 先映射后展平&#xff0c;常用于拆分字符串。 分组 按规则将元素分组为Map结构。 归约 …

数据驱动可视化实战:图表狐精准生成图表的完整数据范式

一、数据输入黄金法则 图表狐 - AI图表生成工具,在线数据可视化要求数据描述必须包含三个核心要素&#xff1a; [主体对象] [量化指标] [维度划分] 错误示例 ❌&#xff1a; "展示各部门销售额对比" 正确示例 ✅&#xff1a; "2023年Q1-Q4各部门销售额&a…

苍穹外卖(1)-部分环境配置(git、数据库)

首先配置git 创建好本地仓库之后 把项目弄到远程仓库里去 先进行提交 &#xff0c;后进行推送 &#xff0c;然后gitee创建一个仓库 把这个url复制好 推送后会出来一个 点击推送&#xff0c;会让你输入gitee账号密码&#xff0c;输入自己的账号密码&#xff0c;就可以连接远程仓…

Ubunut18.04 离线安装MySQL 5.7.35

一、环境准备 1.1 官方下载MySQL5.7.35 完整包 1.2 上传包 & 解压 上传包名称是&#xff1a;mysql-server_5.7.35-1ubuntu18.04_amd64.deb-bundle.tar # 切换到上传目录 cd /home/MySQL # 解压&#xff1a; tar -xvf mysql-server_5.7.35-1ubuntu18.04_amd64.deb-bundle…

Linux(CentOS10) gcc编译

本例子摘自《鸟哥的linux私房菜-基础学习第四版》 21.3 用make进行宏编译 书中的代码在本机器(版本见下&#xff09;编译出错&#xff0c;改正代码后发布此文章&#xff1a; #kernel version: rootlocalhost:~/testmake# uname -a Linux localhost 6.12.0-65.el10.x86_64 #1…

MCP+Blender创建电力塔

MCP&#xff08;Model Context Protocol&#xff09;与Blender的结合是当前AI与3D建模领域的热门技术&#xff0c;它通过协议化的方式让Claude等AI模型直接控制Blender&#xff0c;实现自动化3D建模。 1. 功能与原理 • 核心能力&#xff1a;用户通过自然语言指令&#xff08;…

Qt与C++数据类型转换

本文深入探讨Qt与C中相似但不同的数据类型处理技巧。 一、QString与std::string的相互转换 1. QString → std::string 方法1&#xff1a;使用toStdString()&#xff08;推荐&#xff09; QString qstr "你好&#xff0c;Qt世界"; std::string str qstr.toStdS…

机器学习+EEG熵进行双相情感障碍诊断的综合评估

摘要 双相情感障碍(BD)是一种常见的精神疾病&#xff0c;特点是躁狂或轻躁狂与抑郁交替发作&#xff0c;其严重程度各异&#xff0c;导致准确及时的诊断具有一定的挑战性。EEG的非线性特征被认为是精神障碍的生物标志物&#xff0c;能够反映大脑的非线性动态。尽管已有研究证明…

企业应用集成全析:架构、实践与展望

企业应用集成全析&#xff1a;架构、实践与展望 一、企业应用集成的基本概念1.1 定义1.2 目标 二、企业应用集成的层次架构2.1 数据集成2.2 应用系统集成2.3 业务流程集成​ 三、企业应用集成的关键技术3.1 中间件技术3.2 Web 服务技术​3.3 企业服务总线&#xff08;ESB&#…

【STL】list介绍(附与vector的比较)

文章目录 1.关于list2.使用2.1 list的构造2.2 list 迭代器的使用2.3 list 容量操作2.3.1 size()2.3.2 empty()2.3.3 resize() 2.4 list 元素访问2.4.1 front()2.4.2 back() 2.5 list 修改操作2.5.1 push_front()2.5.2 pop_front()2.5.3 push_back()2.5.4 pop_back()2.5.5 inser…

【Django】教程-12-柱状图

【Django】教程-1-安装创建项目目录结构介绍 【Django】教程-2-前端-目录结构介绍 【Django】教程-3-数据库相关介绍 【Django】教程-4-一个增删改查的Demo 【Django】教程-5-ModelForm增删改查规则校验【正则钩子函数】 【Django】教程-6-搜索框-条件查询前后端 【Django】教程…

SQL:DDL(数据定义语言)和DML(数据操作语言)

目录 什么是SQL&#xff1f; 1. DDL&#xff08;Data Definition Language&#xff0c;数据定义语言&#xff09; 2. DML&#xff08;Data Manipulation Language&#xff0c;数据操作语言&#xff09; DDL和DML的区别 什么是SQL&#xff1f; SQL&#xff08;Structured …

Chrome 135 版本开发者工具(DevTools)更新内容

Chrome 135 版本开发者工具&#xff08;DevTools&#xff09;更新内容 一、性能&#xff08;Performance&#xff09;面板改进 1. 性能面板中的配置文件和函数调用现已显示来源和脚本链接 Performance > Summary&#xff08;性能 > 概览&#xff09;选项卡现在会显示配…