华为OD机试 - 数字排列

题目

屏幕给出1~9任意4个不重复的数字,大家以最快的时间给出这几个数字可拼成的数字从小到大排列位于第N位置的数字,其中N为给出数字中最大的数(如果不到这么多数字,则给出最后一个即可)。
注意:

  1. 2可以当做5来使用 ,反之亦然,但是屏幕不能同时给出2和5;
  2. 6可以当做9来使用 ,反之亦然,但是屏幕不能同时给出6和9

输入

1,4,8,7

输出

41

输入输出说明

给出的最大的数字是8,则能拼接的最大的从小到大的数字为:
1,4,7,8,14,17,18,41,47,48…,第8位为41,所以输出41

输入以逗号分隔的4个1~9的数字组成的字符串。

输出这几个数字可拼成的数字从小到大排列位于第N位置的数字,其中N为给出数字中最大的数。

如果输入的数字不在规定的范围内或有重复,则输出-1

那么第N个数字,即第8个数字为41,输出41。

思路

  1. 1~9
  2. 输入4位数
  3. 不能同时包含2和5,6和9
  4. 2需要当做5来使用,6需要当做9来使用,5需要当做2来使用,9需要当做6来使用
  5. 最快
    如果本题不考虑最快,则可以直接双指针left和right代表下标,依次拼接数字后再排序,然后再输出对应的,但是有最快要求,应该考虑好构建过程就要从小到大

代码

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Scanner;

/**
 * @Desc: 数字排列
 * @Auther:chenShunGuo
 * @Date: 2024/6/7 17:43
 */
public class Test029 {
    /**
     * 2和5可以互换,且不能同时出现
     * 6和9可以互换,且不能同时出现
     */
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        List<Integer> inputList = new ArrayList<>(Arrays.stream(sc.nextLine().split(","))
                .map(Integer::parseInt).sorted().toList());
        // 取输入的最大的数
        int max = inputList.get(3);
        // 校验输入合法性,并按要求重新排序list
        if (!checkInput(inputList)) {
            System.out.println(-1);
            return;
        }
        if (max == 4) {
            System.out.println(max);
            return;
        }
        int size = inputList.size();
        // 排序后从左往右构建,必然是从小到大
        for (int left = 0; left < size; left++) {
            if (inputList.size() >= max) {
                break;
            }
            // 从小到大,先拼接left左边的
            for (int i = 0; i < left; i++) {
                if (inputList.size() >= max) {
                    break;
                }
                buildNum(inputList, inputList.get(left), inputList.get(i));
            }
            // 因为list从小到大,所以left是比right小,拼完左边拼left的右边的数字
            for (int right = left + 1; right < size; right++) {
                if (inputList.size() >= max) {
                    break;
                }
                buildNum(inputList, inputList.get(left), inputList.get(right));
            }
        }
        // 获取组成的数字从小到大排序后第N个数字
        if (inputList.size() < max) {
            System.out.println(inputList.getLast());
        } else {
            System.out.println(inputList.get(max - 1));
        }
        sc.close();
    }

    private static void buildNum(List<Integer> inputList, int left, int right) {
        inputList.add(Integer.valueOf(left + String.valueOf(right)));
    }

    private static boolean checkInput(List<Integer> list) {
        // 如果不足4个数字,则输出-1
        // 输入必须是1~9的数字,排序后如果最小数字小于1,或者最大数字大于9则输出-1
        // 不能同时包含2和5
        // 不能同时包含6和9
        if (list.size() != 4
            || list.get(0) < 1 || list.get(3) > 9
            || (list.contains(2) && list.contains(5))
            || (list.contains(6) && list.contains(9))
        ) {
            return false;
        }

        Map<Integer,Integer> map = new HashMap<>();
        map.put(2,5);
        map.put(5,2);
        map.put(6,9);
        map.put(9,6);
        for (int i = 0; i < 4; i++) {
            int n = list.get(i);
            if (n == 2 || n == 5 || n == 6 || n == 9) {
                list.add(map.get(n));
            }
        }
        // 超过4位,重新排序
        if (list.size() > 4) {
            Collections.sort(list);

        }
        return true;
    }
}

### 华为OD中的斗地主顺子算法华为OD中,针对斗地主游戏内的顺子识别问题,主要考察的是如何通过给定的一系列扑克牌找出所有的顺子组合。此过程涉及到对输入数据的有效处理以及特定逻辑的应用。 #### 数据预处理 为了能够高效地找到所有可能存在的顺子,在开始之前应当先对手中的牌进行排序操作。这一步骤至关重要,因为只有当手中的牌按照大小有序排列之后才能方便后续判断哪些五张连在一起形成合法的顺子[^1]。 #### 寻找顺子的方法 一旦完成了上述准备工作,则可以从第一个元素起始位置出发尝构建长度为5的不同序列,并逐一验证这些候选者是否满足条件——即它们之间是否存在连续递增关系而不含特殊字符&#39;2&#39;[^2]。具体来说: - 对于每一个潜在起点i (0 ≤ i < n - 4),考虑以该点为首的五个相邻项组成的集合; - 判断这个小范围内部份成员间是否有重复现象发生;若有则立即放弃当前选项转而测下一个可能性; - 如果不存在冲突情况的话继续深入探究其余四名成员能否构成严格上升趋势(注意排除掉任何可能出现的最大值‘2’); - 成功匹配一组符合条件的结果后记录下来并准备迎接下一轮挑战直至遍历结束为止。 ```python def find_straights(cards): cards.sort() # 排序 straights = [] for i in range(len(cards)-4): if all(x != &#39;2&#39; and int(x) == int(y)+j-i for j,(x,y) in enumerate(zip(cards[i:i+5],cards[i:i+5][1:]))): straights.append(cards[i:i+5]) return straights ``` 这段Python代码实现了寻找顺子的功能。它首先会对传入的手牌列表`cards`做一次升序整理,接着利用双重循环结构来枚举每一种可能的情况,最后返回一个包含所有成功配对结果的新列表作为输出[^3]。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值