两个字符串间的最短路径问题 - 华为OD统一考试(C卷)

OD统一考试(C卷)

分值: 200分

题解: Java / Python / C++

alt

题目描述

给定两个字符串,分别为字符串A与字符串B。

例如A字符串为ABCABBA,B字符串为CBABAC可以得到下图m*n的二维数组,定义原点为(0, 0),终点为(m, n),水平与垂直的每一条边距离为1,映射成坐标系如下图。

从原点(0, 0)到(0, A)为水平边,距离为1,从(0, A)到(A, C)为垂直边,距离为1;

假设两个字符串同一位置的两个字符相同则可以作一个斜边,如(A, C)到(B, B)最短距离为斜边,距离同样为1。

作出所有的斜边如下图,(0, 0)到(B, B)的距离为 1个水平边 + 1个垂直边 + 1个斜边 = 3。 alt

根据定义可知,原点到终点的最短距离路径如下图红线标记,最短距离为:9 alt

输入描述

空格分割的两个字符串A与字符串B,字符串不为“空串”,字符格式满足正则规则:[A-Z],字符串长度<10000

输出描述

原点到终点的最短距离

示例1

输入:
ABC ABC

输出:
3

示例2

输入:
ABCABBA CBABAC

输出:
9

题解

这是一道典型的动态规划题目。题目要求计算两个字符串之间的最短距离,规定了三种操作:水平边移动、垂直边移动和斜边移动。每一种操作的距离为1。

解题思路

我们可以使用动态规划来解决这个问题。我们可以定义一个二维数组dp,其中dp[r][c]表示从原点到(r, c)点的最短距离。然后,我们可以根据题目规定的操作来更新dp数组。

具体步骤如下:

  1. 初始化dp数组,将所有元素初始化为一个较大的值(表示无穷大),除了dp[0][0]初始化为0。
  2. 初始化边界条件,即从原点到第一行和第一列的最短距离。
  3. 使用双层循环遍历字符串A和B的每个字符,根据题目规定的操作更新dp数组。操作包括水平边移动、垂直边移动和斜边移动。
  4. 最终,dp[m][n]即为从原点到终点的最短距离。

Java

import java.util.Arrays;
import java.util.Scanner;

/**
 * @author code5bug
 */
public class Main {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        String A = scanner.next(), B = scanner.next();
        int m = A.length(), n = B.length();

        // dp[r][c] 为从原点到 (r,c)点的最短距离
        int[][] dp = new int[m + 1][n + 1];
        for (int r = 0; r <= n; r++) Arrays.fill(dp[r], Integer.MAX_VALUE);

        dp[0][0] = 0;
        for (int r = 0; r < m; r++) dp[r + 1][0] = dp[r][0] + 1;
        for (int c = 0; c < n; c++) dp[0][c + 1] = dp[0][c] + 1;

        for (int r = 0; r < m; r++) {
            for (int c = 0; c < n; c++) {
                // 来自从左侧 或 上侧
                dp[r + 1][c + 1] = Math.min(dp[r][c + 1], dp[r + 1][c]) + 1;
                if (A.charAt(r) == B.charAt(c)) {  // 来自从左上角
                    dp[r + 1][c + 1] = Math.min(dp[r + 1][c + 1], dp[r][c] + 1);
                }
            }
        }

        System.out.println(dp[m][n]);
    }
}

Python

from math import inf
A, B = input().split()
cols, rows = len(A), len(B)

# dp[r][c] 为从原点到 (r,c)点的最短距离
dp = [[inf] * (cols + 1) for _ in range(rows + 1)]

# 初始化边界
dp[0][0] = 0


for r in range(0, rows):
    dp[r+1][0] = dp[r][0] + 1
for c in range(0, cols):
    dp[0][c+1] = dp[0][c] + 1

for r in range(rows):
    for c in range(cols):
        dp[r+1][c+1] = min(dp[r+1][c], dp[r][c+1]) + 1
        if A[c] == B[r]:
            dp[r+1][c+1] = min(dp[r+1][c+1], dp[r][c] + 1)

print(dp[rows][cols])
# Python3 AC 96
# PyPy3 AC 100

C++

#include <iostream>
#include <vector>
#include <algorithm>

using namespace std;

int main() {
    string A, B;
    cin >> A >> B;
    int m = A.length(), n = B.length();

    // dp[r][c] 为从原点到(r,c)点的最短距离
    vector<vector<int>> dp(m + 1, vector<int>(n + 1, INT_MAX));
    dp[0][0] = 0;

    for (int r = 0; r < m; r++)
        dp[r + 1][0] = dp[r][0] + 1;

    for (int c = 0; c < n; c++)
        dp[0][c + 1] = dp[0][c] + 1;

    for (int r = 0; r < m; r++) {
        for (int c = 0; c < n; c++) {
            // 来自从左侧或上侧
            dp[r + 1][c + 1] = min(dp[r][c + 1], dp[r + 1][c]) + 1;
            if (A[r] == B[c]) {  // 来自从左上角
                dp[r + 1][c + 1] = min(dp[r + 1][c + 1], dp[r][c] + 1);
            }
        }
    }

    cout << dp[m][n] << endl;

    return 0;
}

相关练习题

alt

🙏整理题解不易, 如果有帮助到您,请给点个赞 ‍❤️‍ 和收藏 ⭐,让更多的人看到。🙏🙏🙏

#面经##校招##春招##秋招##华为#
全部评论
华为od,组内直招,对软硬开发有意向的,可以联系我,Ai算力底座方向,帮改简历。杭州、西安、上海、成都、东莞。c/c /python均可
点赞 回复 分享
发布于 2024-03-19 23:10 浙江

相关推荐

“无名小卒,还是名扬天下?”我知道很多人都不觉得我能走到今天这一步,当然,也包括我自己。在我的人生里,有两部作品刻下了最深的烙印:《斗破苍穹》与《龙族》。它们总被人拿来对照:一边是萧炎的桀骜轻狂,一边是路明非的怯懦衰颓。有人说,天蚕土豆没见过魂天帝,但江南见过真凯撒。我时常觉得,自己就是那个衰小孩路明非。可路明非可以开挂,我不可以;我也无数次幻想过,能拥有萧炎那般年少轻狂的人生,可我没有他与生俱来的逆天天赋。我只是个平庸的普通人,一个看过《斗破苍穹》却开不了挂的路明非,只能一步一步往上爬。从我下定决心找实习的那一刻起,我就给自己定下了目标:“我一定要为字节跳动卖命.jpg”。萧炎有他的三年之约,我有我的两年半之约(其实是一年半)。2024.11.20,科大讯飞的第一封实习offer落进邮箱,我迈出了这场奔赴的第一步。2025.8.18,放弃百度转正的安稳机会,转身走进前路未卜的不确定里。2025.11.14,我选择走进字节跳动,以实习生的身份重新出发。2026.3.25&nbsp;-&nbsp;3.31,一周速通上海飞书,幸遇赏识我的伯乐,斩获Special&nbsp;Offer。被告知面试通过的那一刻,我的内心无比平静,就像这个offer本就该属于我。不是侥幸,是应得的。这一路,有人看轻过我的出身,不相信我能走到这里;也有人在我看不见前路的时候,替我举过灯。没有他们的鼓励与支撑,就没有今天站在这里的我。我看到了自强不息的激荡,那是一个双非的伟大乐章!我是雨夜迈巴赫,我要开启属于我的新篇章了。
在看牛客的本杰明很勇...:真心祝贺l总 我永远的偶像 我滴神
春招至今,你收到几个面试...
点赞 评论 收藏
分享
努力的小明a:项目看着很眼熟,施磊老师吧,我也学的这个😋我当时是把rpc框架做成了一个分布式网盘,这是一个项目,然后muduo库做成集群即时通讯,又用QT做了个交互的客户端,这样又一个项目,然后一个轻量redis,一个CAD,总共四个项目,投了三个月就今天2月份一个小厂Qt offer,然后后面想开了,Qt啥的都能干,这个月get了个北京大厂的offer,做java后端,人生就是这么魔幻,现在就在去北京入职的路上
点赞 评论 收藏
分享
评论
3
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务