【华为OD题库-106】全排列-java

题目

给定一个只包含大写英文字母的字符串S,要求你给出对S重新排列的所有不相同的排列数。如:S为ABA,则不同的排列有ABA、AAB、BAA三种。
解答要求
时间限制:5000ms,内存限制:100MB
输入描述
输入一个长度不超过10的字符串S,确保都是大写的。
输出描述
输出S重新排列的所有不相同的排列数(包含自己本身)。
示例1:
输入
ABA
输出
3
示例2:
输入
ABCDEFGHHA
输出
907200

思路

求含重复元素的排列总数,两种方法(数学法效率远高于dfs):

  1. dfs列举所有排列,得到总数,只需要总数,不需要具体组合,具体思路详见:【JAVA-排列组合】一个套路速解排列组合题
  2. 数学公式法

比如输入数据为:ABCDDDA
不考虑重复数据,总的排列数为:res=7!
统计重复元素出现次数:A出现2次,D出现3次,其他出现1次
所以最后结果为:res=7!/(2!*3!)=420

备注:数学法可以这样理解,不考虑重复,那么n个元素的排列方法为n!,再去重,比如某个元素有3个,在不去重时,它重复了3!次,去重时直接除掉它就可以了

题解

package hwod;

import java.util.*;

public class ArrangeInOrder {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        String str = sc.nextLine();
        System.out.println(arrangeInOrder(str));
        System.out.println(arrangeInOrder2(str));
    }

    private static int res;

    private static int arrangeInOrder(String str) {
        final char[] chars = str.toCharArray();
        Arrays.sort(chars);
        int[] used = new int[chars.length];
        dfs(chars, used, 0);
        return res;
    }

    private static void dfs(char[] chars, int[] used, int len) {
        if (len == chars.length - 1) {
            //确定到倒数第二位即可
            res++;
            return;
        }

        for (int i = 0; i < chars.length; i++) {
            if (i > 0 && chars[i] == chars[i - 1] && used[i - 1] == 0) continue;
            if (used[i] == 1) continue;
            used[i] = 1;
            dfs(chars, used, len + 1);
            used[i] = 0;
        }
    }

    //方案二:数学法
    private static int arrangeInOrder2(String str) {
        Map<Character, Integer> map = statisticsCnt(str);
        int size = str.length();
        long res = getFactorialNum(size);
        int divide = 1;
        for (Character key : map.keySet()) {
            divide *= getFactorialNum(map.get(key));
        }
        return (int) (res / divide);
    }

    private static Map<Character, Integer> statisticsCnt(String string) {
        Map<Character, Integer> map = new HashMap<>();
        for (int i = 0; i < string.length(); i++) {
            map.put(string.charAt(i), map.getOrDefault(string.charAt(i), 0) + 1);
        }
        return map;
    }

    /**
     * @param n
     * @return 计算阶乘
     */
    private static long getFactorialNum(long n) {
        if (n == 1) return 1;
        return n * getFactorialNum(n - 1);
    }
}

推荐

如果你对本系列的其他题目感兴趣,可以参考华为OD机试真题及题解(JAVA),查看当前专栏更新的所有题目。

说明

本专栏所有文章均为原创,欢迎转载,请注明文章出处:https://blog.youkuaiyun.com/qq_31076523/article/details/134176793。百度和各类采集站皆不可信,搜索请谨慎鉴别。技术类文章一般都有时效性,本人习惯不定期对自己的博文进行修正和更新,因此请访问出处以查看本文的最新版本。

### 华为暑期实习机试题库及相关资料 华为暑期实习生招聘通常会涉及在线笔试环节,其考察范围广泛,涵盖了基础编程能力、算法设计以及实际业务场景的应用。以下是关于华为暑期实习机试的相关信息: #### 一、机试题型概述 华为OD机试试题主要分为两类:一类是基于具体应用场景的综合题目;另一类则是经典的算法与数据结构问题。例如,在2024年的OD机试中出现了诸如消消乐游戏逻辑实现[^1]和栈的数据合并操作等问题[^2]。 这些题目不仅考验考生的基础编码技能,还注重对复杂业务需求的理解能力和解决实际工程问题的能力。因此,准备此类考试时应重点复习以下几个方面: - **经典算法**:如排序(快速排序)、查找(二分法)等。 - **数据结构应用**:包括但不限于链表、堆栈、队列的操作及其变种形式。 - **组合优化问题**:像资源分配或者路径规划这类需要权衡多个因素的任务。 #### 二、推荐学习方向与工具书 为了更好地应对即将到来的华为暑期实习机试,建议采取如下措施来提升自己的竞争力: 1. **熟悉主流编程语言特性** 掌握至少一种常用开发环境下的高级程序设计方法论非常重要。对于Java而言,除了了解面向对象的基本概念外,还需要深入理解多线程处理机制、异常管理框架等内容。 2. **强化实战演练经验积累** 可以参考历年公开发布的真题集锦来进行针对性训练。比如,“九宫格全排列”、“扑克牌消除逻辑”都是很好的练习素材[^3]。另外还可以尝试模拟真实项目中的小型功能模块开发工作流,从而锻炼整体架构思维模式。 3. **利用网络平台获取最新动态资讯** 加入一些专门讨论技术话题的兴趣小组微信群/QQ群组,定期关注官方博客公告栏更新情况,及时掌握最新的考核趋势变化特点。 #### 三、样例代码展示 下面给出一段简单的Java示例代码用于演示如何完成一个基本的功能单元——判断给定字符串是否为回文串: ```java public class PalindromeChecker { public static boolean isPalindrome(String s){ int left=0; int right=s.length()-1; while(left<right){ char cLeft = s.charAt(left); char cRight = s.charAt(right); if(!Character.isLetterOrDigit(cLeft)){ left++; continue; } if(!Character.isLetterOrDigit(cRight)){ right--; continue; } if(Character.toLowerCase(cLeft)!= Character.toLowerCase(cRight)) return false; left++; right--; } return true; } } ``` 此函数实现了忽略大小写及非字母数字字符干扰条件下检测输入序列正反读取一致性的核心逻辑。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值