【双机位A卷】华为OD笔试之【哈希表】双机位A-跳房子I【Py/Java/C++/C/JS/Go六种语言】【欧弟算法】全网注释最详细分类最全的华子OD真题题解

可上 欧弟OJ系统 练习华子OD、大厂真题

绿色聊天软件戳 od1441了解算法冲刺训练(备注【CSDN】否则不通过)

文章目录

相关推荐阅读

题目练习网址:【哈希表】双机位A-跳房子I

题目描述与示例

题目描述

跳房子,也叫跳飞机,是一种世界性的儿童游戏。 游戏参与者需要分多个回合按顺序跳到第1格直到房子的最后一格

跳房子的过程中,可以向前跳,也可以向后跳。

假设房子的总格数是count,小红每回合可能连续跳的步教都放在数组steps中,请问数组中是否有一种步数的组合,可以让小红两个回合 跳到最后一格? 如果有,请输出索引和最小的步数组合

注意:

  • 数组中的步数可以重复,但数组中的元素不能重复使用。
  • 提供的数据保证存在满足题目要求的组合,且索引和最小的步数组合是唯一的

输入描述

第一行输入为每回合可能连续跳的步数,它是整数数组类型。

第二行输入为房子总格数count,它是int整数类型。

输出描述

返回索引和最小的满足要求的步数组合(顺序保持steps中原有顺序)

备注

  • count ≤ 1000
  • 0 ≤ steps.length ≤ 5000
  • -100000000 ≤ steps ≤ 100000000

示例一

输入

Plain 复制代码
[1,4,5,2]
7

输出

Plain 复制代码
[5,2]

示例二

输入

Plain 复制代码
[-1,2,4,9,6]
8

输出

Plain 复制代码
[-1,9]

解题思路

注意,本题和LeetCode1、两数之和几乎完全一致。区别在于需要输出索引和最小的两个数字。

本题关于哈希表的部分已经强调过很多遍了,因此不再花费篇幅进行介绍。

所以本篇题解主要介绍一些同学们在这一道题里面比较困惑的一些细节处理上的问题

输入与输出

本题的输入其他很多题目不同的地方在于,输入的字符串虽然能够表示一个数组,但是其前后包含了中括号"[]"

如果我们想要根据逗号","对原字符串进行切割的话,会得到第一个元素和最后一个元素分别包含"[""]"

为了避免这种情况,我们可以对input()得到的字符串利用切片进行掐头去尾的操作,即

python 复制代码
s = input()[1:-1]

再结合内置函数map()和方法split(),我们可以得到数字列表nums

python 复制代码
nums = list(map(int, input()[1:-1].split(",")))

同理,本题答案的输出也需要包含中括号"[]"

假设最终答案数组ans = [0, 1],如果我们直接输出ans,即

python 复制代码
print(ans)

会得到以下输出

Shell 复制代码
[0, 1]

注意到逗号","的后面的包含空格" "的。但这是错误的输出结果,会得0分!

仔细观察题目要求的输出,逗号后面不应该包含空格。正确的输出应该是

Shell 复制代码
[0,1]

为了能够正确得到上述结果,我们不得不使用格式化字符串的方式进行输出,即

python 复制代码
print(f"[{ans[0]},{ans[1]}]")

这再一次提醒我们(在其他题目中也有所体现),ACM模式的判题机制是相当严格的

题目要求你的答案输出和期望答案输出完全一致

多一个空格少一个空格,中英文标点符号不分,都会导致整道题目直接得0分

索引和最小值的初始值设置

本题和LeetCode1、两数之和最大的区别在于,本题可能存在多组和为target的数对。

题目要求我们需要输出索引和最小的两个数字。所以我们必须在原版题目的基础上加上这个过程。

首先我们考虑一个非常简单的问题。

如果要求我们在不使用内置函数min()max()的前提下,找到一个数组nums中的最小值,我们会如何做?

这个问题非常简单,我们可以这样完成:

  1. 设置一个初始值ans表示数组中可能的最小值。
  2. 用循环遍历整个数组nums,如果发现某个数字num比初始值更小的时候,则我们将ans替换为这个数字。
  3. 退出循环后,最终的ans即为答案。
python 复制代码
nums = [4, 1, 3, 2]
ans = nums[0]        # 或者ans = math.inf

for num in nums:
    if ans > num:
        ans = num

在初始化ans的时候,我们有两种策略:

  • ans越大越好,必须要大过nums中的最小值。所以可以设置ans = inf
  • ans是一个可能的答案。数组中的每一个元素都是可能的答案,所以可以设置ans = nums[0]

回到这道题本身,我们要找到索引和最小的两个数字,可以设置一个变量min_idx_sum来表示这个索引和。

那么很多同学困惑的地方在于,下面的代码为什么是这样初始化变量min_idx_sum的。

python 复制代码
min_idx_sum = len(nums)*2

由于后面我们希望能够找到更小的索引和,所以我们可以把min_idx_sum初始化为一个较大值。

显然,对于长度n = len(nums)而言的数组,最大的两个索引是n-1n-2,它们的和小于n*2

当我们设置min_idx_sum的初始值为len(nums)*2时,其实跟设置inf或者len(nums)*2-2是一样的。

PS:在此我也希望大家能够举一反三。

如果某个题目问的是寻找最大值且元素都是正数的话,那么我们就可以设置ans初始值为0或者-inf了。

一定要学会变通。

寻找索引和最小值

接下来的任务就是,如何把索引和最小值的问题,加入到经典题目LeetCode1、两数之和中。

如果不加限制,其算法过程如下:

  1. 遍历nums的每一个索引i和对应的数字num
  2. 计算剩余数字rest_num = target-num是否位于哈希表hash_dic中,若在则得到答案ans
  3. 将当前数字num作为key,当前索引i作为value,将键值对(num, i)储存入哈希表hash_dic

其代码如下

python 复制代码
for i, num in enumerate(nums):
    rest_num = target-num
    if rest_num in hash_dic:
        ans = [rest_num, num]
    hash_dic[num] = i

但由于我们希望寻找到的是索引和最小值。假设在原数组中存在两个相同的数字,我们在哈希表中储存的索引i,必然希望它是更早出现的那个而不是更晚出现的那个。

所以在更新哈希表的时候,需要额外多加一个判断:如果num已经位于哈希表中了,则不进行更新。

修改后代码如下

python 复制代码
for i, num in enumerate(nums):
    rest_num = target-num
    if rest_num in hash_dic:
        ans = [rest_num, num]
    if num not in hash_dic:
        hash_dic[num] = i

另外,只有当当前两数索引和,小于全局索引和的时候,我们才需要更新答案ans

当前数字num的索引为i,剩余数字rest_num的索引为hash_dic[rest_num]。故修改后代码如下

python 复制代码
for i, num in enumerate(nums):
    rest_num = target-num
    if rest_num in hash_dic:
        if min_idx_sum > hash_dic[rest_num] + i:
            min_idx_sum = hash_dic[rest_num] + i
            ans = [rest_num, num]
    if num not in hash_dic:
        hash_dic[num] = i

特别注意,当min_idx_sum > hash_dic[rest_num] + i成立后,既要修改答案列表ans,也要修改全局的索引最小和min_idx_sum为当前更新的hash_dic[rest_num] + i

综上就构成了本题的核心算法逻辑了。

*不能在第一次找到答案就退出

在和同学们交流的过程中,有同学提出了以下解法。

python 复制代码
for i, num in enumerate(nums):
    rest_num = target-num
    if rest_num in hash_dic:
        ans = [rest_num, num]
        break
    if num not in hash_dic:
        hash_dic[num] = i

即在第一次找到符合要求的答案时就退出。

同学的理由非常简单,由于遍历是从左到右进行的,那么找到的第一组自然就是索引和最小的情况。

但这是一个比较典型的、过于想当然的思路。

我们可以非常容易地举出反例。考虑例子

python 复制代码
target = 9
nums = [1, 5, 6, 3, 8]

显然有两组索引[0, 4][2, 3](注意是索引而非元素本身)能使得nums中对应的两个数的和为target

如果是从左往右遍历,我们会先找到索引对[2, 3],但显然[2, 3]的和会小于[0, 4]

一旦在找到[2, 3]时就提前退出,那么将找不到正确结果[0, 4]

代码

Python

python 复制代码
# 欢迎来到「欧弟算法 - 华为OD全攻略」,收录华为OD题库、面试指南、八股文与学员案例!
# 地址:https://www.odalgo.com
# 华为OD机试刷题网站:https://www.algomooc.com
# 添加微信 278166530 获取华为 OD 笔试真题题库和视频

# 题目:2024E/2025C/双机位A-跳房子I
# 分值:100
# 作者:许老师-闭着眼睛学数理化
# 算法:哈希表
# 代码看不懂的地方,请直接在群上提问


# 输入步数列表,注意需要去除最开头和最末尾的中括号
nums = list(map(int, input()[1:-1].split(",")))
target = int(input())

# 初始化索引和最小值的取值,这里可以取inf,也可以取nums长度乘2
min_idx_sum = len(nums)*2
# 构建哈希表,储存每一种步数首次出现的下标
hash_dic = dict()
# 初始化答案列表
ans = [0, 0]

# 遍历nums
for i, num in enumerate(nums):
    # 计算target-num的值rest_num
    rest_num = target-num
    # 若rest_num位于哈希表中,说明其曾经出现过
    # rest_num和num相加等于所需要的结果target
    if rest_num in hash_dic:
        # 若此时min_idx_sum大于两者下标和
        # 则更新min_idx_sum和ans
        if min_idx_sum > hash_dic[rest_num] + i:
            min_idx_sum = hash_dic[rest_num] + i
            # 注意题目要求两个元素保持原有顺序
            ans = [rest_num, num]
    # 若num不位哈希表中,说明它第一次出现,记录其下标
    # 由于我们希望两数的下标和尽可能小
    # 故对于重复出现的数字,只记录其第一次出现的下标即可
    if num not in hash_dic:
        hash_dic[num] = i


# 输出答案,注意要按照格式输出,逗号后面不带空格,不能直接输出ans
print(f"[{ans[0]},{ans[1]}]")

Java

java 复制代码
import java.util.*;

public class Main {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        String input = scanner.nextLine();
        input = input.substring(1, input.length() - 1); // Remove square brackets
        String[] numStrings = input.split(",");
        int[] nums = new int[numStrings.length];

        for (int i = 0; i < nums.length; i++) {
            nums[i] = Integer.parseInt(numStrings[i].trim());
        }

        int target = scanner.nextInt();
        int minIdxSum = nums.length * 2;
        HashMap<Integer, Integer> hashDic = new HashMap<>();
        int[] ans = new int[2];

        for (int i = 0; i < nums.length; i++) {
            int num = nums[i];
            int restNum = target - num;

            if (hashDic.containsKey(restNum)) {
                if (minIdxSum > hashDic.get(restNum) + i) {
                    minIdxSum = hashDic.get(restNum) + i;
                    ans[0] = restNum;
                    ans[1] = num;
                }
            }

            if (!hashDic.containsKey(num)) {
                hashDic.put(num, i);
            }
        }

        System.out.println("[" + ans[0] + "," + ans[1] + "]");
    }
}

C++

cpp 复制代码
#include <iostream>
#include <vector>
#include <unordered_map>
#include <sstream>
using namespace std;

int main() {
    string input;
    getline(cin, input);
    input = input.substr(1, input.length() - 2); // Remove square brackets
    istringstream iss(input);
    vector<int> nums;
    string numStr;
    
    while (getline(iss, numStr, ',')) {
        nums.push_back(stoi(numStr));
    }
    
    int target;
    cin >> target;
    int minIdxSum = nums.size() * 2;
    unordered_map<int, int> hashDic;
    vector<int> ans(2);
    
    for (int i = 0; i < nums.size(); i++) {
        int num = nums[i];
        int restNum = target - num;
        
        if (hashDic.find(restNum) != hashDic.end()) {
            if (minIdxSum > hashDic[restNum] + i) {
                minIdxSum = hashDic[restNum] + i;
                ans[0] = restNum;
                ans[1] = num;
            }
        }
        
        if (hashDic.find(num) == hashDic.end()) {
            hashDic[num] = i;
        }
    }
    
    cout << "[" << ans[0] << "," << ans[1] << "]" << endl;
    
    return 0;
}

C

c 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>

/* =========================
 * 简易哈希表(拉链法) num -> firstIndex
 * 仅记录每个数第一次出现的下标
 * ========================= */
typedef struct Entry {
    int key;               /* 数值 */
    int val;               /* 最早下标 */
    struct Entry* next;
} Entry;

typedef struct {
    Entry** buckets;
    int cap;               /* 桶数 */
} HashMap;

static unsigned int hash_int(int x) {
    unsigned int u = (unsigned int)x;
    u ^= u >> 16;
    u *= 0x7feb352dU;
    u ^= u >> 15;
    u *= 0x846ca68bU;
    u ^= u >> 16;
    return u;
}

static void hinit(HashMap* h, int cap) {
    h->cap = cap;
    h->buckets = (Entry**)calloc(cap, sizeof(Entry*));
}

static Entry* hfind(HashMap* h, int key) {
    unsigned int idx = hash_int(key) % h->cap;
    for (Entry* e = h->buckets[idx]; e; e = e->next) {
        if (e->key == key) return e;
    }
    return NULL;
}

static void hput_if_absent(HashMap* h, int key, int val) {
    unsigned int idx = hash_int(key) % h->cap;
    for (Entry* e = h->buckets[idx]; e; e = e->next) {
        if (e->key == key) return; /* 已存在则不更新(保持最早下标) */
    }
    Entry* e = (Entry*)malloc(sizeof(Entry));
    if (!e) exit(1);
    e->key = key; e->val = val;
    e->next = h->buckets[idx];
    h->buckets[idx] = e;
}

static void hfree(HashMap* h) {
    for (int i = 0; i < h->cap; ++i) {
        Entry* e = h->buckets[i];
        while (e) { Entry* n = e->next; free(e); e = n; }
    }
    free(h->buckets);
}

/* 去掉字符串首尾空白 */
static void trim(char* s) {
    int n = (int)strlen(s);
    int i = 0, j = n - 1;
    while (i <= j && isspace((unsigned char)s[i])) i++;
    while (j >= i && isspace((unsigned char)s[j])) j--;
    if (i > j) { s[0] = '\0'; return; }
    if (i > 0) memmove(s, s + i, (size_t)(j - i + 1));
    s[j - i + 1] = '\0';
}

/* 读取一整行,去掉行尾换行 */
static int read_line(char* buf, int cap) {
    if (!fgets(buf, cap, stdin)) return 0;
    size_t L = strlen(buf);
    while (L && (buf[L - 1] == '\n' || buf[L - 1] == '\r')) buf[--L] = '\0';
    return 1;
}

int main(void) {
    char line[1 << 16];   /* 第一行数组 */
    char line2[1 << 16];  /* 第二行 target */

    if (!read_line(line, sizeof(line))) {
        /* 没有输入,按无解输出 */
        printf("[0,0]\n");
        return 0;
    }

    /* 提取方括号中的内容 */
    trim(line);
    char *lb = strchr(line, '[');
    char *rb = strrchr(line, ']');
    if (!lb || !rb || rb <= lb + 1) {
        /* 形如 [] 或格式异常 -> 视为空数组 */
        printf("[0,0]\n");
        return 0;
    }
    *rb = '\0';
    char *inner = lb + 1;

    /* 解析逗号分隔的整数到动态数组 */
    int cap = 16, n = 0;
    int *nums = (int*)malloc(sizeof(int) * cap);
    if (!nums) return 1;

    char *tok = strtok(inner, ",");
    while (tok) {
        trim(tok);
        if (*tok) {
            long v = strtol(tok, NULL, 10);
            if (n == cap) {
                cap = (cap < 1024) ? cap * 2 : cap + 1024;
                int *t = (int*)realloc(nums, sizeof(int) * cap);
                if (!t) { free(nums); return 1; }
                nums = t;
            }
            nums[n++] = (int)v;
        }
        tok = strtok(NULL, ",");
    }

    /* 读取 target(第二行),若没有则默认 0 */
    int target = 0;
    if (read_line(line2, sizeof(line2))) {
        trim(line2);
        if (*line2) target = (int)strtol(line2, NULL, 10);
    }

    /* 遍历寻找下标和最小的解 */
    int minIdxSum = n * 2;
    int ansA = 0, ansB = 0;   /* 默认无解输出 [0,0](与原实现一致) */

    HashMap mp; hinit(&mp, 100003); /* 取素数容量降低冲突 */

    for (int i = 0; i < n; ++i) {
        int num = nums[i];
        int rest = target - num;

        Entry* e = hfind(&mp, rest);
        if (e) {
            if (e->val + i < minIdxSum) {
                minIdxSum = e->val + i;
                ansA = rest;
                ansB = num;
            }
        }
        /* 只记录"第一次"出现的下标 */
        if (!hfind(&mp, num)) {
            hput_if_absent(&mp, num, i);
        }
    }

    printf("[%d,%d]\n", ansA, ansB);

    hfree(&mp);
    free(nums);
    return 0;
}

Node JavaScript

javascript 复制代码
const fs = require('fs');
const data = fs.readFileSync(0, 'utf8').trim().split(/\r?\n/);

// 解析第一行数组
let line = (data[0] || '').trim();
if (line.startsWith('[') && line.endsWith(']')) {
  line = line.slice(1, -1); // 去掉方括号
}
const nums = line.length
  ? line.split(',').map(s => parseInt(s.trim(), 10))
  : [];

// 解析 target(第二行)
const target = data.length >= 2 ? parseInt(data[1].trim(), 10) : 0;

let minIdxSum = nums.length * 2;
const firstIndex = new Map(); // num -> 最早下标
let ansA = 0, ansB = 0;       // 默认无解时输出 [0,0]

// 遍历数组
for (let i = 0; i < nums.length; i++) {
  const num = nums[i];
  const rest = target - num;

  if (firstIndex.has(rest)) {
    const j = firstIndex.get(rest);
    if (j + i < minIdxSum) {
      minIdxSum = j + i;
      ansA = rest;
      ansB = num;
    }
  }
  // 只记录最早下标
  if (!firstIndex.has(num)) {
    firstIndex.set(num, i);
  }
}

console.log(`[${ansA},${ansB}]`);

Go

go 复制代码
package main

import (
        "bufio"
        "fmt"
        "os"
        "strconv"
        "strings"
)


func main() {
        in := bufio.NewReader(os.Stdin)

        // 第一行:数组
        line1, _ := in.ReadString('\n')
        line1 = strings.TrimSpace(line1)
        if strings.HasPrefix(line1, "[") && strings.HasSuffix(line1, "]") {
                line1 = line1[1 : len(line1)-1] // 去掉方括号
        }
        nums := []int{}
        if len(strings.TrimSpace(line1)) > 0 {
                parts := strings.Split(line1, ",")
                for _, p := range parts {
                        v, _ := strconv.Atoi(strings.TrimSpace(p))
                        nums = append(nums, v)
                }
        }

        // 第二行:target
        line2, _ := in.ReadString('\n')
        line2 = strings.TrimSpace(line2)
        target, _ := strconv.Atoi(line2)

        minIdxSum := len(nums) * 2
        firstIndex := make(map[int]int) // num -> 最早下标
        ansA, ansB := 0, 0              // 默认无解 [0,0]

        for i, num := range nums {
                rest := target - num
                if j, ok := firstIndex[rest]; ok {
                        if j+i < minIdxSum {
                                minIdxSum = j + i
                                ansA, ansB = rest, num
                        }
                }
                // 只记录第一次出现
                if _, ok := firstIndex[num]; !ok {
                        firstIndex[num] = i
                }
        }

        fmt.Printf("[%d,%d]\n", ansA, ansB)
}

时空复杂度

时间复杂度:O(``N``)。仅需一次遍历数组。

空间复杂度:O(``N``)。哈希表所占空间。


华为OD算法/大厂面试高频题算法练习冲刺训练

  • 华子OD算法/大厂面试高频题算法冲刺训练目前开始常态化报名!目前已服务1000+同学成功上岸!

  • 课程讲师为全网200w+粉丝编程博主@吴师兄学算法 以及小红书头部编程博主@闭着眼睛学数理化

  • 90+天陪伴式学习,100+直播课时,300+动画图解视频,500+LeetCode经典题,500+华为OD真题/大厂真题,还有简历修改、模拟面试、陪伴小群、资深HR对接将为你解锁

  • 可上全网独家的欧弟OJ系统练习华子OD、大厂真题

  • 可查看链接OD真题汇总(持续更新)

  • 绿色聊天软件戳 od1441或了解更多

相关推荐
老前端的功夫21 小时前
Vue2中key的深度解析:Diff算法的性能优化之道
前端·javascript·vue.js·算法·性能优化
xiaoxiongip66621 小时前
假设两个设备在不同网段,网关怎么设置才能通呢
网络·爬虫·python·https·智能路由器
Full Stack Developme21 小时前
java.net.http 包详解
java·http·.net
淀粉肠kk21 小时前
【C++】map和set的使用
c++
逻极1 天前
Scikit-learn 实战:15 分钟构建生产级中国房价预测模型
python·机器学习·scikit-learn
行板Andante1 天前
AttributeError: ‘super‘ object has no attribute ‘sklearn_tags‘解决
人工智能·python·sklearn
tryCbest1 天前
Python基础之爬虫技术(一)
开发语言·爬虫·python
husterlichf1 天前
逻辑回归以及python(sklearn)详解
python·逻辑回归·sklearn
AI小云1 天前
【Numpy数据运算】数组间运算
开发语言·python·numpy
daidaidaiyu1 天前
Spring BeanPostProcessor接口
java·spring