当前位置: 首页 > news >正文

华为OD机试真题——最长的顺子(2025B卷:100分)Java/python/JavaScript/C++/C语言/GO六种最佳实现

在这里插入图片描述

2025 B卷 100分 题型

本文涵盖详细的问题分析、解题思路、代码实现、代码详解、测试用例以及综合分析;
并提供Java、python、JavaScript、C++、C语言、GO六种语言的最佳实现方式!

本文收录于专栏:《2025华为OD真题目录+全流程解析/备考攻略/经验分享》

华为OD机试真题《最长的顺子》:


目录

    • 题目名称:最长的顺子
      • 题目描述
    • Java
      • 问题分析
      • 解题思路
      • 代码实现
      • 代码详细解析
      • 示例测试
      • 综合分析
    • python
      • 问题分析
      • 解题思路
      • 代码实现
      • 代码详细解析
      • 示例测试
      • 综合分析
    • JavaScript
      • 问题分析
      • 解题思路
      • 代码实现
      • 代码详细解析
      • 示例测试
      • 综合分析
    • C++
      • 问题分析
      • 解题思路
      • 代码实现
      • 代码详细解析
      • 示例测试
        • 示例1输入:
        • 示例2输入(无顺子):
        • 示例3输入(完整顺子):
      • 综合分析
    • C语言
      • 问题分析
      • 解题思路
      • 代码实现
      • 代码详细解析
      • 示例测试
        • 示例1输入:
        • 示例2输入(无顺子):
        • 示例3输入(最长顺子):
      • 综合分析
    • GO
      • 问题分析
      • 解题思路
      • 代码实现
      • 代码详细解析
      • 示例测试
        • 示例1输入:
        • 示例2输入(无顺子):
        • 示例3输入(最长顺子):
      • 综合分析


题目名称:最长的顺子


知识点:字符串、动态规划/滑动窗口、逻辑处理
时间限制:1秒
空间限制:256MB
语言限制:不限


题目描述

输入当前手牌和已出牌,计算对手可能组成的最长顺子(连续5-12张牌,不含2和大小王)。若存在多个最长顺子,输出牌面最大的那个,否则返回"NO-CHAIN"。

规则说明

  1. 顺子定义:连续递增的牌序列,范围从3到A(3<4<5<6<7<8<9<10<J<Q<K<A),不含2、B(小王)、C(大王)。
  2. 输入格式
    • 第一行为当前手牌(如 3-3-4-5-A
    • 第二行为已出牌(如 A-4-5-6-7
  3. 输出格式:最长顺子,若长度相同则输出牌面最大的(如 9-10-J-Q-K-A 优于 5-6-7-8-9)。
  4. 数据约束:每张牌初始有4张(除大小王),总牌数为54张。

输入示例

3-3-4-4-5-A-5-6-2-8-3-9-10-Q-7-K-J-10-B  
A-4-5-8-8-10-C-6-7-8  

输出示例

9-10-J-Q-K-A  

Java

问题分析

我们需要找到对手可能组成的最长顺子。顺子由3到A的连续牌组成,不含2和大小王。输出最长顺子,若存在多个相同长度的,选择牌面最大的那个。若无法组成,返回"NO-CHAIN"。

解题思路

  1. 统计剩余牌数:根据手牌和已出牌,计算每个有效牌的剩余数量。
  2. 滑动窗口遍历:从每个可能的起始牌开始,尽可能向右扩展窗口,直到遇到剩余不足的牌。
  3. 选择最优结果:记录最长窗口,若长度相同则选择末尾最大的窗口。

代码实现

import java.util.*;public class Main {public static void main(String[] args) {Scanner scanner = new Scanner(System.in);// 读取输入的两行并分割成牌列表String handInput = scanner.nextLine().trim();String playedInput = scanner.nextLine().trim();List<String> allCards = new ArrayList<>();if (!handInput.isEmpty()) {allCards.addAll(Arrays.asList(handInput.split("-")));}if (!playedInput.isEmpty()) {allCards.addAll(Arrays.asList(playedInput.split("-")));}// 有效牌的顺序和集合String[] order = {"3", "4", "5", "6", "7", "8", "9", "10", "J", "Q", "K", "A"};Set<String> validCards = new HashSet<>(Arrays.asList(order));// 统计每个牌的出现次数Map<String, Integer> countMap = new HashMap<>();for (String card : allCards) {if (validCards.contains(card)) {countMap.put(card, countMap.getOrDefault(card, 0) + 1);}}// 计算每个牌的剩余次数int[] remain = new int[order.length];for (int i = 0; i < order.length; i++) {String card = order[i];int used = countMap.getOrDefault(card, 0);remain[i] = 4 - used; // 每张牌最多4张}// 滑动窗口遍历寻找最长顺子int maxLen = 0;int bestStart = 0;int bestEnd = 0;for (int i = 0; i < order.length; i++) {if (remain[i] <= 0) continue; // 起始牌必须可用int currentEnd = i;// 向右扩展窗口,直到遇到不可用的牌for (int j = i; j < order.length; j++) {if (remain[j] <= 0) break;currentEnd = j;}int currentLen = currentEnd - i + 1;// 更新最长且牌面最大的顺子if (currentLen >= 5) {if (currentLen > maxLen || (currentLen == maxLen && currentEnd > bestEnd)) {maxLen = currentLen;bestStart = i;bestEnd = currentEnd;}}}// 输出结果if (maxLen >= 5) {List<String> result = new ArrayList<>();for (int i = bestStart; i <= bestEnd; i++) {result.add(order[i]);}System.out.println(String.join("-", result));} else {System.out.println("NO-CHAIN");}}
}

代码详细解析

  1. 输入处理:读取手牌和已出牌,分割为列表。
  2. 有效牌集合:定义合法的顺子牌和顺序。
  3. 统计次数:遍历所有牌,统计每个有效牌的出现次数。
  4. 剩余计算:每种牌最多4张,剩余次数为4 - 已用次数
  5. 滑动窗口
    • 外层遍历每个起始点i,若该牌剩余可用则继续。
    • 内层扩展窗口j,直到遇到剩余为0的牌,记录窗口结束位置。
    • 更新最长且最大的顺子。
  6. 结果输出:根据窗口生成顺子字符串或输出“NO-CHAIN”。

示例测试

示例1输入

3-3-4-4-5-A-5-6-2-8-3-9-10-Q-7-K-J-10-B  
A-4-5-8-8-10-C-6-7-8  

输出9-10-J-Q-K-A
解析:剩余牌中9到A的连续序列最长且牌面最大。

示例2输入(无法形成顺子):

3-3-3-3  
4-4-4-4  

输出NO-CHAIN

示例3输入(最长顺子):

3-4-5-6-7-8-9-10-J-Q-K-A  
(空已出牌)

输出3-4-5-6-7-8-9-10-J-Q-K-A

综合分析

  1. 时间复杂度:O(n²),n为有效牌数量(12),双重循环遍历。
  2. 空间复杂度:O(n),存储剩余次数和牌顺序。
  3. 优势:滑动窗口确保找到最优解,处理高效。
  4. 适用场景:适用于题目约束的小规模输入,快速找到最长顺子。

python

问题分析

我们需要找到对手可能组成的最长顺子。顺子由3到A的连续牌组成,不含2和大小王。输出最长顺子,若存在多个相同长度的,选择牌面最大的那个。若无法组成,返回"NO-CHAIN"。


解题思路

  1. 输入处理:将手牌和已出牌合并,统计每张牌的出现次数。
  2. 剩余牌计算:每张牌最多4张,减去已出现的次数得到剩余数量。
  3. 滑动窗口遍历:从每个可能的起点出发,寻找最长的连续可用牌序列。
  4. 最优顺子选择:记录最长且最大的顺子。

代码实现

def main():# 读取输入的两行数据hand = input().strip().split('-') if input().strip() != '' else []played = input().strip().split('-') if input().strip() != '' else []all_cards = hand + played# 定义有效牌的顺序(从3到A)order = ["3", "4", "5", "6", "7", "8", "9", "10", "J", "Q", "K", "A"]valid_cards = set(order)  # 用于快速判断是否为有效牌# 统计每张牌的使用次数count 

相关文章:

  • 洪水危险性评价与风险防控全攻略:从HEC-RAS数值模拟到ArcGIS水文分析,一键式自动化工具实战,助力防洪减灾与应急管理
  • 力扣面试150题--二叉搜索树迭代器
  • 海量数据查询加速:Presto、Trino、Apache Arrow 实战指南
  • 第五十二节:增强现实基础-简单 AR 应用实现
  • Odoo 自动化规则全面深度解析
  • 《仿盒马》app开发技术分享-- 地图选点(端云一体)
  • Python爬虫实战:研究Selenium框架相关技术
  • 大数据下HashMap 扩容优化方案及选择
  • 哈希表day5
  • 【C++】给定数据长度n,采样频率f,频率分辨率是多少?
  • day37打卡
  • 微信小程序进阶第2篇__事件类型_冒泡_非冒泡
  • 精益数据分析(86/126):Parse.ly的转型启示——从用户增长到商业变现的艰难跨越
  • kali切换为中文
  • Golang 的协程调度小结
  • 原子操作(C++)
  • 初等数论--Garner‘s 算法
  • crash常用命令
  • JavaScripts API(应用程序编程接口)
  • 提问:鲜羊奶是解决育儿Bug的补丁吗?
  • 网站目的及功能定位/设计网站排行榜前十名
  • vi设计网站大全/百度推广的效果
  • 外贸电商做俄罗斯市场网站/竞价推广工作内容
  • 常州专业网站建设公司咨询/东莞公司seo优化
  • 兰州网新公司/厦门seo排名优化方式
  • 徐州做网站最好的公司/seo领导屋