华为OD机试 - 几何平均值最大子数(Python/JS/C/C++ 2024 E卷 200分)

news2024/11/30 15:39:15

在这里插入图片描述

华为OD机试 2024E卷题库疯狂收录中,刷题点这里

专栏导读

本专栏收录于《华为OD机试真题(Python/JS/C/C++)》。

刷的越多,抽中的概率越大,私信哪吒,备注华为OD,加入华为OD刷题交流群,每一题都有详细的答题思路、详细的代码注释、3个测试用例、为什么这道题采用XX算法、XX算法的适用场景,发现新题目,随时更新,全天CSDN在线答疑。

一、题目描述

从一个长度为N的正整数数组numbers中找出长度至少为L且几何平均值Q最大的子数组,并输出其位置和大小。(K个数的几何平均值为K个数的乘积的K次方根)

若有多个子数组的几何平均值均为最大值Q,则输出长度最小的子数组。

若有多个长度最小的子数组的几何平均值均为最大值,则输出最前面的子数组。

二、输入描述

第1行输入N、L:

N表示numbers的大小(1 <= N <= 100000)

L表示子数组的最小长度(1 <= L <= N)

之后N行表示numbers中的N个数,每行一个(1 <= numbers[i] <= 10^9)

三、输出描述

输出子数组的位置(从0开始计数)和大小,中间用一个空格隔开。

备注
用例保证除几何平均值为最大值的子数组外,其他子数组的几何平均值至少比最大值小10^-10倍。

四、测试用例

测试用例1:

1、输入

3 2
2
2
3

2、输出

1 2

3、说明

长度至少为2的子数组共有三个,分别是{2,2}, {2,3}, {2,3},其中{2,3}的几何平均值最大,故输出其位置1和长度2。

测试用例2:

1、输入

10 2
0.2
0.2
0.1
0.2
0.2
0.2
0.1
0.2
0.2
0.2

2、输出

0 2

3、说明

所有长度为2的子数组及其几何平均值:

索引0-1: [0.2, 0.2],GM = 0.2
索引1-2: [0.2, 0.1],GM ≈ 0.141
索引2-3: [0.1, 0.2],GM ≈ 0.141
索引3-4: [0.2, 0.2],GM = 0.2
索引4-5: [0.2, 0.2],GM = 0.2
索引5-6: [0.2, 0.1],GM ≈ 0.141
索引6-7: [0.1, 0.2],GM ≈ 0.141
索引7-8: [0.2, 0.2],GM = 0.2
索引8-9: [0.2, 0.2],GM = 0.2

最大几何平均值为0.2,对应多个子数组。选择最早的子数组,即从索引0开始的子数组,因此输出0 2。

五、解题思路

1、问题分析

我们需要在一个长度为N的正数数组numbers中找出长度至少为L且几何平均值最大的子数组,并输出其起始位置和长度。如果存在多个满足条件的子数组,我们需要按照以下优先级选择:

  1. 几何平均值最大的子数组。
  2. 如果有多个几何平均值相同,选择长度最小的子数组。
  3. 如果仍有多个,选择最前面的子数组。

由于N的范围较大(1 ≤ N ≤ 100,000),需要设计一个高效的算法。

2、具体步骤:

  1. 读取输入:读取N和L,接着读取N个正数。
  2. 对数转换:对每个数取自然对数,存储在一个数组中。
  3. 前缀和计算:计算对数数组的前缀和,以便快速计算任意子数组的对数和。
  4. 滑动窗口遍历:遍历所有长度为L的子数组,计算其对数和,记录最大对数和及其起始位置。
  5. 输出结果:输出起始位置和子数组长度L。

六、Python算法源码

import math
import sys

def main():
    # 读取输入
    input = sys.stdin.read().split()
    index = 0
    
    # 读取数组大小N和子数组最小长度L
    N = int(input[index])
    index += 1
    L = int(input[index])
    index += 1
    
    # 初始化数组并计算对数值
    log_values = []
    for _ in range(N):
        num = float(input[index])
        index += 1
        log_values.append(math.log(num))  # 计算每个数的自然对数
    
    # 计算前缀和数组
    prefix_sums = [0.0] * (N + 1)
    for i in range(1, N + 1):
        prefix_sums[i] = prefix_sums[i - 1] + log_values[i - 1]
    
    # 初始化最大对数平均值及对应子数组的起始位置和长度
    max_log_average = -math.inf
    result_start_index = 0
    result_length = L
    
    # 初始化最小前缀和及其索引
    min_prefix = 0.0
    min_prefix_index = 0
    
    # 遍历前缀和数组,寻找最大对数平均值的子数组
    for i in range(L, N + 1):
        # 更新最小前缀和为[0, i - L]区间的最小前缀和
        if prefix_sums[i - L] < min_prefix:
            min_prefix = prefix_sums[i - L]
            min_prefix_index = i - L
        
        # 计算当前对数平均值
        current_log_average = (prefix_sums[i] - min_prefix) / L
        
        # 更新最大对数平均值及对应的子数组位置和长度
        if current_log_average > max_log_average:
            max_log_average = current_log_average
            result_start_index = min_prefix_index
            result_length = L
        elif math.isclose(current_log_average, max_log_average, abs_tol=1e-10):
            # 如果对数平均值相同,选择长度更小的子数组
            if L < result_length:
                result_start_index = min_prefix_index
                result_length = L
            # 如果长度相同,选择起始位置更靠前的子数组(无需操作,因为遍历顺序已保证)
    
    # 输出结果
    print(result_start_index, result_length)

if __name__ == "__main__":
    main()

七、JavaScript算法源码

// 引入readline模块以读取输入
const readline = require('readline');

const rl = readline.createInterface({
    input: process.stdin,
    output: process.stdout
});

let input = [];
rl.on('line', function(line){
    input = input.concat(line.trim().split(' '));
}).on('close', function(){
    let index = 0;
    
    // 读取数组大小N和子数组最小长度L
    const N = parseInt(input[index++]);
    const L = parseInt(input[index++]);
    
    // 初始化数组并计算对数值
    const logValues = [];
    for(let i = 0; i < N; i++){
        const num = parseFloat(input[index++]);
        logValues.push(Math.log(num)); // 计算每个数的自然对数
    }
    
    // 计算前缀和数组
    const prefixSums = new Array(N + 1).fill(0);
    for(let i = 1; i <= N; i++){
        prefixSums[i] = prefixSums[i - 1] + logValues[i - 1];
    }
    
    // 初始化最大对数平均值及对应子数组的起始位置和长度
    let maxLogAverage = -Infinity;
    let resultStartIndex = 0;
    let resultLength = L;
    
    // 初始化最小前缀和及其索引
    let minPrefix = 0.0;
    let minPrefixIndex = 0;
    
    // 遍历前缀和数组,寻找最大对数平均值的子数组
    for(let i = L; i <= N; i++){
        // 更新最小前缀和为[0, i - L]区间的最小前缀和
        if(prefixSums[i - L] < minPrefix){
            minPrefix = prefixSums[i - L];
            minPrefixIndex = i - L;
        }
        
        // 计算当前对数平均值
        const currentLogAverage = (prefixSums[i] - minPrefix) / L;
        
        // 更新最大对数平均值及对应的子数组位置和长度
        if(currentLogAverage > maxLogAverage){
            maxLogAverage = currentLogAverage;
            resultStartIndex = minPrefixIndex;
            resultLength = L;
        } else if(Math.abs(currentLogAverage - maxLogAverage) < 1e-10){
            // 如果对数平均值相同,选择长度更小的子数组
            if(L < resultLength){
                resultStartIndex = minPrefixIndex;
                resultLength = L;
            }
            // 如果长度相同,选择起始位置更靠前的子数组(无需操作,因为遍历顺序已保证)
        }
    }
    
    // 输出结果
    console.log(resultStartIndex + " " + resultLength);
});

八、C算法源码

#include <stdio.h>
#include <math.h>
#include <float.h>

int main(){
    int N, L;
    // 读取数组大小N和子数组最小长度L
    scanf("%d %d", &N, &L);
    
    double log_values[N];
    // 读取数组元素并计算对数值
    for(int i = 0; i < N; i++){
        double num;
        scanf("%lf", &num);
        log_values[i] = log(num); // 计算每个数的自然对数
    }
    
    // 计算前缀和数组
    double prefix_sums[N + 1];
    prefix_sums[0] = 0.0;
    for(int i = 1; i <= N; i++){
        prefix_sums[i] = prefix_sums[i - 1] + log_values[i - 1];
    }
    
    // 初始化最大对数平均值及对应子数组的起始位置和长度
    double max_log_average = -DBL_MAX;
    int result_start_index = 0;
    int result_length = L;
    
    // 初始化最小前缀和及其索引
    double min_prefix = 0.0;
    int min_prefix_index = 0;
    
    // 遍历前缀和数组,寻找最大对数平均值的子数组
    for(int i = L; i <= N; i++){
        // 更新最小前缀和为[0, i - L]区间的最小前缀和
        if(prefix_sums[i - L] < min_prefix){
            min_prefix = prefix_sums[i - L];
            min_prefix_index = i - L;
        }
        
        // 计算当前对数平均值
        double current_log_average = (prefix_sums[i] - min_prefix) / L;
        
        // 更新最大对数平均值及对应的子数组位置和长度
        if(current_log_average > max_log_average){
            max_log_average = current_log_average;
            result_start_index = min_prefix_index;
            result_length = L;
        }
        else if(fabs(current_log_average - max_log_average) < 1e-10){
            // 如果对数平均值相同,选择长度更小的子数组
            if(L < result_length){
                result_start_index = min_prefix_index;
                result_length = L;
            }
            // 如果长度相同,选择起始位置更靠前的子数组(无需操作,因为遍历顺序已保证)
        }
    }
    
    // 输出结果
    printf("%d %d\n", result_start_index, result_length);
    
    return 0;
}

九、C++算法源码

#include <bits/stdc++.h>
using namespace std;

int main(){
    ios::sync_with_stdio(false);
    cin.tie(0);
    
    int N, L;
    // 读取数组大小N和子数组最小长度L
    cin >> N >> L;
    
    vector<double> log_values(N);
    // 读取数组元素并计算对数值
    for(int i = 0; i < N; i++){
        double num;
        cin >> num;
        log_values[i] = log(num); // 计算每个数的自然对数
    }
    
    // 计算前缀和数组
    vector<double> prefix_sums(N + 1, 0.0);
    for(int i = 1; i <= N; i++){
        prefix_sums[i] = prefix_sums[i - 1] + log_values[i - 1];
    }
    
    // 初始化最大对数平均值及对应子数组的起始位置和长度
    double max_log_average = -1e308; // 接近负无穷
    int result_start_index = 0;
    int result_length = L;
    
    // 初始化最小前缀和及其索引
    double min_prefix = 0.0;
    int min_prefix_index = 0;
    
    // 遍历前缀和数组,寻找最大对数平均值的子数组
    for(int i = L; i <= N; i++){
        // 更新最小前缀和为[0, i - L]区间的最小前缀和
        if(prefix_sums[i - L] < min_prefix){
            min_prefix = prefix_sums[i - L];
            min_prefix_index = i - L;
        }
        
        // 计算当前对数平均值
        double current_log_average = (prefix_sums[i] - min_prefix) / L;
        
        // 更新最大对数平均值及对应子数组的位置和长度
        if(current_log_average > max_log_average){
            max_log_average = current_log_average;
            result_start_index = min_prefix_index;
            result_length = L;
        }
        else if(abs(current_log_average - max_log_average) < 1e-10){
            // 如果对数平均值相同,选择长度更小的子数组
            if(L < result_length){
                result_start_index = min_prefix_index;
                result_length = L;
            }
            // 如果长度相同,选择起始位置更靠前的子数组(无需操作,因为遍历顺序已保证)
        }
    }
    
    // 输出结果
    cout << result_start_index << " " << result_length << "\n";
    
    return 0;
}


🏆下一篇:华为OD机试真题 - 简易内存池(Python/JS/C/C++ 2024 E卷 200分)

🏆本文收录于,华为OD机试真题(Python/JS/C/C++)

刷的越多,抽中的概率越大,私信哪吒,备注华为OD,加入华为OD刷题交流群,每一题都有详细的答题思路、详细的代码注释、3个测试用例、为什么这道题采用XX算法、XX算法的适用场景,发现新题目,随时更新,全天CSDN在线答疑。

在这里插入图片描述

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

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

相关文章

uniapp固定document.title标题

由于业务中需要将h5的标题固定 但是uniapp没有对应的接口 所以使用Object.defineProperty拦截set方法来实现 代码也很简单 在App.vue的onLaunch加上就行了 onLaunch: function() {document.title 固定标题;Object.defineProperty(document, title, {set() {return false;}});…

USB 3.0?USB 3.1?USB 3.2?怎么区分?

还记得小白刚接触电脑的时候&#xff0c;电脑普及的USB接口大部分是USB 2.0&#xff0c;还有少部分USB 1.0的&#xff08;现在基本上找不到了&#xff09;。 当时的电脑显示器&#xff0c;可能00后的小伙伴都没见过&#xff0c;它们大概长这样&#xff1a; 当时小白以为电脑最…

C++ 算法学习——1.6 差分算法与二维差分算法

一维差分算法概述&#xff1a; 差分算法是一种用于计算序列中相邻元素之间差值的技术。在C中&#xff0c;STL&#xff08;标准模板库&#xff09;提供了std::adjacent_difference函数来实现差分算法。 std::adjacent_difference函数&#xff1a; std::adjacent_difference函数位…

基于MindSpore实现CycleGAN壁画修复

基于MindSpore实现CycleGAN壁画修复_哔哩哔哩_bilibili 本案例运行需要较大内存&#xff0c;建议在Ascend/GPU上运行。 模型介绍 模型简介 CycleGAN(Cycle Generative Adversarial Network) 即循环对抗生成网络&#xff0c;来自论文 Unpaired Image-to-Image Translation us…

指针赋值or常数赋值

int main (){int a 10;int b ;b a;int *c &a;int *d c; } 常数 a,b赋值&#xff1a; 都是将存储的值&#xff08;10&#xff09;赋值给别人。 指针赋值也是类似的&#xff1a; 指针存储的值&#xff08;&a&#xff09;为地址&#xff0c;就是把c指向的地址赋值给…

C语言 | Leetcode C语言题解之第458题可怜的小猪

题目&#xff1a; 题解&#xff1a; int poorPigs(int buckets, int minutesToDie, int minutesToTest){int base minutesToTest / minutesToDie 1;int pigs ceil(log(buckets)/log(base));return pigs; }

24-10-2-读书笔记(二十二)-《契诃夫文集》(一)上([俄] 契诃夫 [译] 汝龙)啊!真想生活。

文章目录 《契诃夫文集》&#xff08;一&#xff09;上&#xff08;[俄] 契诃夫 [译] 汝龙 &#xff09;早期生活——塔甘罗格&#xff08;人物家庭简介&#xff09;学生时期——莫斯科&#xff08;写作与学习&#xff09;流浪时期——哈萨林&#xff08;游历与流浪&#xff09…

VCSEL驱动电路

1.1 驱动电路 发射端可用MOS管控制VCSEL二极管负极方式发出脉冲光(正极对地)&#xff0c;具体作用过程如下&#xff1a; Step 1: MOS管断开, C2 电容充电(左侧HV)&#xff1b; Step 2: 信号控制MOS管打开&#xff1b; Step 3: MOS管打开后, C2电容左侧电压降为0V, 右侧变为…

当你系统有多个模块的时候,要设计统一入口页了。

本次给大家带来一批系统统一入口页的界面&#xff0c;这次都是科技感十足的界面。入口页将所有系统集中展示出来&#xff0c;并辅以其他设计元素进行氛围烘托&#xff0c;别看简单&#xff0c;但真的不好设计。

免费送源码:Java+B/S+MySQL 基于springboot网上书店管理系统 计算机毕业设计原创定制

基于springboot网上书店管理系统 摘 要 网上书店管理系统采用B/S结构、java开发语言、以及Mysql数据库等技术。系统主要分为管理员和用户两部分&#xff0c;管理员管理主要功能包括&#xff1a;首页、网站管理&#xff08;轮播图、网站公告&#xff09;人员管理&#xff08;管…

MATLAB - 机械臂手眼标定(眼在手内) - 估计安装在机器人上的移动相机的姿态

系列文章目录 前言 本示例展示了如何为装有手眼构型摄像头的机械臂或机械手执行和验证手眼校准。 一、概述 执行手眼校准有助于操作配备末端执行器&#xff08;简称 “手”&#xff09;的机械臂&#xff0c;该末端执行器依赖于摄像头提供的视觉数据。一旦完成了眼在手外的校准&…

数据结构阶段测试2的一点小补充

数据结构阶段测试2的一点小补充 1.已知⼩根堆为8,15,10,21,34,16,12&#xff0c;删除关键字8之后需重建堆&#xff0c;最后的叶⼦ 节点为() A. 34 B. 21 C. 16 D. 12 解题思路 向下调整算法删除堆顶元素 &#x1f4a1; 答案&#xff1a;C 删除堆顶元素的思路&#xff1a; …

环境对象刺激,recordnunber,记忆柱,记忆柱群,主注意对象,目的对象,状态中枢,奖惩预期,思维等等之间的联系

我们清醒状态下&#xff0c;随时都有目的&#xff0c;目的控制影响着我们思想行为的方向。目的是用对象来表征的&#xff08;目的对象&#xff09;&#xff0c;对象之所以能够表征目的&#xff0c;是因为对象能够被&#xff08;状态性&#xff09;赋值&#xff08;任何赋值都是…

UE4 材质学习笔记03(翻书(Flipbook)动画/环境混合)

一.FlipBook Animation 如果你想让游戏以每秒30帧的速度运行&#xff0c;所有内容都必须在33毫秒内渲染出来&#xff0c; 如果你想让游戏以每秒60帧的速度运行的话&#xff0c;必须在16毫秒内。 所以当一个效果需要很多细节的时候&#xff0c;往往会离线创建它&#xff0c;然…

【Qt】控件概述(4)—— 输出类控件

输出类控件 1. QLineEdit——单行输入框2. QTextEdit——多行输入框3. QComboBox——下拉框4. QSpinBox——微调框5. QDateEdit && QTimeEdit && QDateTimeEdit6 QDial——旋钮7. QSlider——滑动条 1. QLineEdit——单行输入框 QLineEdit是一个单行的输入框&…

BUU刷题-Pwn-shanghai2018_baby_arm(ARM_ROP_csu_init,ARM架构入门)

解题思路&#xff1a; 泄露或修改内存数据&#xff1a; 堆地址&#xff1a;无需栈地址&#xff1a;无需libc地址&#xff1a;无需BSS段地址&#xff1a;无需 劫持程序执行流程&#xff1a;ARM_ROP && mprotect函数(运行内存权限修改) && [[ARM_ROP_csu_init]…

【AI自然语言处理应用】通义晓蜜CCAI

通义晓蜜CCAI-对话分析AIO 对话分析AIO&#xff0c;即对话分析all-in-one API&#xff0c;是基于深度调优的对话大模型&#xff0c; 为营销服类产品提供智能化升级所需的生成式摘要总结、质检、分析等能力的官方应用。 面向对象&#xff1a;开发者、自研企业、传统呼叫中心采购…

02 nth_element 与第k小

题目&#xff1a; 方案一&#xff1a;sort排序 #include<bits/stdc.h> using namespace std;int main() {int n;int k;cin>>n>>k;int a[n]{0};for(int i0;i<n;i){cin>>a[i];}sort(a,an); cout<<a[k]<<endl;}方案二&#xff1a;…

【机器学习(十一)】糖尿病数据集分类预测案例分析—XGBoost分类算法—Sentosa_DSML社区版

文章目录 一、XGBoost算法二、Python代码和Sentosa_DSML社区版算法实现对比(一) 数据读入和统计分析(二)数据预处理(三)模型训练与评估(四)模型可视化 三、总结 一、XGBoost算法 关于集成学习中的XGBoost算法原理&#xff0c;已经进行了介绍与总结&#xff0c;相关内容可参考【…

leetcode面试题17.04:消失的数字(C语言版)

思路1 先排序&#xff0c;再依次查找&#xff0c;如果下一个值不等于前一个1&#xff0c;那么下一个值就是消失数字。 时间复杂度分析&#xff1a;冒泡排序的时间复杂度为O(N^2)&#xff0c;qsort排序时间复杂度为O(N*logN)。因此该思路不可行。 思路2 求和0到N&#xff0c;再减…