LeetCode 3272.统计好整数的数目:枚举+排列组合+哈希表

news2025/4/13 9:07:26

【LetMeFly】3272.统计好整数的数目:枚举+排列组合+哈希表

力扣题目链接:https://leetcode.cn/problems/find-the-count-of-good-integers/

给你两个  整数 n 和 k 。

如果一个整数 x 满足以下条件,那么它被称为 k 回文 整数 。

  • x 是一个 回文整数 。
  • x 能被 k 整除。

如果一个整数的数位重新排列后能得到一个 k 回文整数 ,那么我们称这个整数为 整数。比方说,k = 2 ,那么 2020 可以重新排列得到 2002 ,2002 是一个 k 回文串,所以 2020 是一个好整数。而 1010 无法重新排列数位得到一个 k 回文整数。

请你返回 n 个数位的整数中,有多少个  整数。

注意 ,任何整数在重新排列数位之前或者之后 都不能 有前导 0 。比方说 1010 不能重排列得到 101 。

 

示例 1:

输入:n = 3, k = 5

输出:27

解释:

部分好整数如下:

  • 551 ,因为它可以重排列得到 515 。
  • 525 ,因为它已经是一个 k 回文整数。

示例 2:

输入:n = 1, k = 4

输出:2

解释:

两个好整数分别是 4 和 8 。

示例 3:

输入:n = 5, k = 6

输出:2468

 

提示:

  • 1 <= n <= 10
  • 1 <= k <= 9

解题方法:枚举+排列组合+哈希表

我们可以枚举所有长度为 n n n的数字回文字符串。对于一个回文串,如果其对应整数能够被 k k k整除,则这个字符串的所有排列都能被 k k k整除。

问题就转化为了以下三个:

  1. 如何枚举所有长度为 n n n的数字回文串?

    我们可以枚举字符串的前半部分,然后反转得到后半部分并拼接,就得到了回文字符串。

    具体的,我们可以从 [ 1 0 ⌊ n − 1 2 ⌋ , 1 0 ⌊ n − 1 2 ⌋ + 1 ) [10^{\lfloor\frac{n-1}2\rfloor}, 10^{\lfloor\frac{n-1}2\rfloor+1}) [102n1,102n1+1)枚举前半字符串,若 n n n为奇数则包含中间位置元素。

  2. 一个“k回文整数字符串”有多少个排列?多少个字符串重新排列后能得到这个字符串?

    我们分别统计字符串中每个数字出现了多少次,假设字符串长度为 n n n

    首先计算第一位有多少种可能,第一位不能为 0 0 0,因此方案数为 n − t i m e s [ 0 ] n-times[0] ntimes[0]

    接着计算其他位,其他的 n − 1 n-1 n1个数字可以随便排列,其他位的方案数为 ( n − 1 ) ! (n-1)! (n1)!

    由于其中可能存在重复元素,例如 121 121 121中有两个 1 1 1,这两个 1 1 1谁在前谁在后无法区分,只能有一种方案,所以要除以 2 ! 2! 2!

    总的方案数为: ( n − t i m e s [ 0 ] ) × ( n − 1 ) ! Π i = 0 9 t i m e s [ i ] \frac{(n-times[0])\times (n-1)!}{\Pi_{i=0}^{9}times[i]} Πi=09times[i](ntimes[0])×(n1)!

  3. 如何保证每个排列不会被重复统计?

    假设我已经统计过 1221 1221 1221的所有全排列了,那么我遇到 2112 2112 2112时就不能再统计 2112 2112 2112的所有全排列了,因为 1221 1221 1221 2112 2112 2112的全排列时相同的。

    如何做到?其实我们只需要使用一个哈希表,记录字符串 s s s排序后的字符串是否出现过就好了。

时空复杂度分析

  • 时间复杂度 O ( 1 0 m × n log ⁡ n ) O(10^{m}\times n\log n) O(10m×nlogn),其中 m = ⌊ n − 1 2 ⌋ m=\lfloor\frac{n-1}2\rfloor m=2n1
  • 空间复杂度 O ( 1 0 m × n ) O(10^m\times n) O(10m×n)

AC代码

C++
/*
 * @Author: LetMeFly
 * @Date: 2025-04-12 07:51:15
 * @LastEditors: LetMeFly.xyz
 * @LastEditTime: 2025-04-12 09:25:01
 * @Description: AC,21.21%,93.94%
 */
#if defined(_WIN32) || defined(__APPLE__)
#include "_[1,2]toVector.h"
#endif

/*
1 -> 1-9
2 -> 1-9
3 -> 10->99
4 -> 10->99
5 -> 100->999
6 -> 100->999

n -> [10^((n-1)/2-1), 10^((n-1)/2)-1)
*/
typedef long long ll;

class Solution {
private:
    int k;
    unordered_set<string> visited;
    vector<int> factor;
    int times[10];

    void initFactor(int n) {
        factor.resize(n + 1);
        factor[0] = 1;
        for (int i = 1; i <= n; i++) {
            factor[i] = factor[i - 1] * i;
        }
    }

    bool ifVisited(string s) {
        sort(s.begin(), s.end());
        if (visited.count(s)) {
            return true;
        }
        visited.insert(s);
        return false;
    }
    
    bool isOk(string& s) {
        ll val = stoll(s);
        // printf("%s: %d\n", s.c_str(), val % k == 0);  // *****
        return val % k == 0;
    };

    ll calc(string& s) {
        memset(times, 0, sizeof(times));
        for (char c : s) {
            times[c - '0']++;
        }
        ll ans = (s.size() - times[0]) * factor[s.size() - 1];
        for (int i = 0; i < 10; i++) {
            ans /= factor[times[i]];
        }
        return ans;
    }
public:
    ll countGoodIntegers(int n, int k) {
        initFactor(n);
        this->k = k;
        ll ans = 0;
        for (int start = pow(10, (n - 1) / 2), end = start * 10; start < end; start++) {
            string prefix = to_string(start), suffix = prefix.substr(0, prefix.size() - n % 2);
            reverse(suffix.begin(), suffix.end());
            string thisNum = prefix + suffix;
            if (isOk(thisNum) && !ifVisited(thisNum)) {  // 注意ifVisited会将thisNum加入哈希表的话记得先判断isOk再判断ifVisited
                // printf("ans: %lld, calc(%s): %lld, ans = ans + calc(%s) = %lld\n", ans, thisNum.c_str(), calc(thisNum), thisNum.c_str(), ans + calc(thisNum));  // ****
                ans += calc(thisNum);
            }
        }
        return ans;
    }
};
Python
'''
Author: LetMeFly
Date: 2025-04-12 09:44:25
LastEditors: LetMeFly.xyz
LastEditTime: 2025-04-12 10:52:36
Description: 中间终中断了下
'''
from collections import Counter

class Solution:
    def isOk(self, s: str) -> bool:
        return int(s) % self.k == 0
    
    def ifVisited(self, s: str) -> bool:
        s = ''.join(sorted(s))
        if s in self.visited:
            return True
        self.visited.add(s)
        return False
    
    def calc(self, s: str) -> int:
        times = Counter(s)
        ans = (len(s) - times['0']) * self.factor[len(s) - 1]
        for _, val in times.items():
            ans //= self.factor[val]
        return ans

    def countGoodIntegers(self, n: int, k: int) -> int:
        self.k = k
        self.factor = [1] * (n + 1)
        for i in range(1, n + 1):
            self.factor[i] = self.factor[i - 1] * i
        self.visited = set()
        base = pow(10, (n - 1) // 2)
        ans = 0
        for i in range(base, base * 10):
            prefix = str(i)
            s = prefix + prefix[::-1][n % 2:]
            if self.isOk(s) and not self.ifVisited(s):
                ans += self.calc(s)
        return ans

Java
/*
 * @Author: LetMeFly
 * @Date: 2025-04-12 10:53:42
 * @LastEditors: LetMeFly.xyz
 * @LastEditTime: 2025-04-12 11:13:08
 */
import java.util.Set;
import java.util.HashSet;
// import java.lang.StringBuilder;  // 默认自动导入 无需手动导入
import java.util.Arrays;

class Solution {
    private int k;
    private int[] factor;
    private Set<String> visited = new HashSet<>();

    private boolean isOk(String s) {
        return Long.parseLong(s) % k == 0;
    }

    private boolean ifVisited(String s) {
        char[] array = s.toCharArray();
        Arrays.sort(array);
        String sorted = new String(array);
        return !visited.add(sorted);
    }

    private long calc(String s) {
        int[] times = new int[10];
        for (char c : s.toCharArray()) {
            times[c - '0']++;
        }
        long ans = (s.length() - times[0]) * factor[s.length() - 1];
        for (int i = 0; i < 10; i++) {
            ans /= factor[times[i]];
        }
        return ans;
    }

    public long countGoodIntegers(int n, int k) {
        this.k = k;
        factor = new int[n + 1];
        factor[0] = 1;
        for (int i = 1; i <= n; i++) {
            factor[i] = factor[i - 1] * i;
        }
        long ans = 0;
        for (int from = (int)Math.pow(10, (n - 1) / 2), to = from * 10; from < to; from++) {
            String prefix = String.valueOf(from);
            String suffix = new StringBuilder(prefix).reverse().substring(n % 2);
            String s = prefix + suffix;
            if (isOk(s) && !ifVisited(s)) {
                ans += calc(s);
            }
        }
        return ans;
    }
}

/*
API:

java 子字符串
反转字符串
整数转字符串
字符串拼接
字符串转整数
*/
Go
/*
 * @Author: LetMeFly
 * @Date: 2025-04-12 11:14:05
 * @LastEditors: LetMeFly.xyz
 * @LastEditTime: 2025-04-12 11:40:36
 * @Description: AC,25.00%,75.00%,一遍过
 */
package main

import (
    "math"
    "strconv"
    "sort"
    "strings"
)

type solution3273 struct{
    n, k int
    factor []int
    visited map[string]bool
}

func init3273(n int, k int) *solution3273 {
    ans := &solution3273{
        n: n,
        k: k,
        factor: make([]int, n + 1),
        visited : map[string]bool{},
    }
    ans.factor[0] = 1
    for i := 1; i <= n; i++ {
        ans.factor[i] = ans.factor[i - 1] * i
    }
    return ans
}

func (t* solution3273) isOk(s string) bool {
    val, _ := strconv.Atoi(s)
    return val % t.k == 0
}

func (t* solution3273) ifVisited(s string) bool {
    array := strings.Split(s, "")
    sort.Strings(array)
    s = strings.Join(array, "")
    if t.visited[s] {
        return true
    }
    t.visited[s] = true
    return false
}

func (t* solution3273) calc(s string) (ans int64) {
    times := [10]int{}
    for i, _ := range s {
        times[s[i] - '0']++
    }
    ans = int64(t.n - times[0]) * int64(t.factor[t.n - 1])
    for _, v := range times {
        ans /= int64(t.factor[v])
    }
    return
}

func (t* solution3273) getFullS(prefix string) string {
    suffix := []byte(prefix)
    if t.n % 2 == 1 {
        suffix = suffix[:len(suffix) - 1]
    }
    for i := 0; i < len(suffix) / 2; i++ {
        suffix[i], suffix[len(suffix) - i - 1] = suffix[len(suffix) - i - 1], suffix[i]
    }
    return prefix + string(suffix)
}

func countGoodIntegers(n int, k int) (ans int64) {
    solution := init3273(n, k)
    from := int(math.Pow10((n - 1) / 2))
    to := from * 10
    for i := from; i < to; i++ {
        s := solution.getFullS(strconv.Itoa(i))
        if solution.isOk(s) && !solution.ifVisited(s) {
            ans += solution.calc(s)
        }
    }
    return
}

同步发文于CSDN和我的个人博客,原创不易,转载经作者同意后请附上原文链接哦~

千篇源码题解已开源

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2333707.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

国标GB28181视频平台EasyCVR如何搭建汽车修理厂远程视频网络监控方案

一、背景分析 近年我国汽车保有量持续攀升&#xff0c;与之相伴的汽车保养维修需求也逐渐提高。随着社会经济的发展&#xff0c;消费者对汽车维修服务质量的要求越来越高&#xff0c;这使得汽车维修店的安全防范与人员管理问题面临着巨大挑战。 多数汽车维修店分布分散&#…

PostIn安装及入门教程

PostIn是一款国产开源免费的接口管理工具&#xff0c;包含项目管理、接口调试、接口文档设计、接口数据MOCK等模块&#xff0c;支持常见的HTTP协议、websocket协议等&#xff0c;支持免登陆本地接口调试&#xff0c;本文将介绍如何快速安装配置及入门使用教程。 1、安装 私有…

spring cloud微服务API网关详解及各种解决方案详解

微服务API网关详解 1. 核心概念 定义&#xff1a;API网关作为微服务的统一入口&#xff0c;负责请求路由、认证、限流、监控等功能&#xff0c;简化客户端与后端服务的交互。核心功能&#xff1a; 路由与转发&#xff1a;将请求分发到对应服务。协议转换&#xff1a;HTTP/HTTP…

最新版PhpStorm超详细图文安装教程,带补丁包(2025最新版保姆级教程)

目录 前言 一、PhpStorm最新版下载 二、PhpStorm安装 三、PhpStorm补丁 四、运行PhpStorm 前言 PhpStorm 是 JetBrains 公司推出的 专业 PHP 集成开发环境&#xff08;IDE&#xff09;&#xff0c;专为提升 PHP 开发效率设计。其核心功能包括智能代码补全、实时语法错误检…

linux kernel arch 目录介绍

一&#xff1a;arch 目录 二&#xff1a;常用arch

ES6变量声明:let、var、const全面解析

一、引言 ECMAScript 6&#xff08;简称 ES6&#xff09;的发布为 JavaScript 带来了许多革命性的变化&#xff0c;其中变量声明方式的更新尤为重要。let、var和const成为开发者日常编码中频繁使用的关键字。 本文将深入解析这三种声明方式的核心特性、区别及最佳实践&#xff…

Linux 入门八:Linux 多进程

一、概述 1.1 什么是进程&#xff1f; 在 Linux 系统中&#xff0c;进程是程序的一次动态执行过程。程序是静态的可执行文件&#xff0c;而进程是程序运行时的实例&#xff0c;系统会为其分配内存、CPU 时间片等资源。例如&#xff0c;输入 ls 命令时&#xff0c;系统创建进程…

单调栈 —— 1.基本概念与核心算法

1. 基本概念 1.1 知识预备 在理解单调栈之前&#xff0c;我们需要先掌握两个基础概念&#xff1a;栈&#xff08;Stack&#xff09; 和 单调性&#xff08;Monotonicity&#xff09;。 什么是栈&#xff08;Stack&#xff09; 栈是一种**后进先出&#xff08;LIFO, Last-In…

工程师 - 场效应管分类

What Are the Different Types of FETs? Pulse Octopart Staff Jul 31, 2021 Field effect transistors (FETs) are today’s workhorses for digital logic, but they enjoy plenty of applications outside of digital integrated circuits, everything from motor driver…

Debezium报错处理系列之第128篇:增量快照报错java.lang.OutOfMemoryError: Java heap space

Debezium报错处理系列之第128篇:增量快照报错java.lang.OutOfMemoryError: Java heap space 一、完整报错二、错误原因三、解决方法Debezium从入门到精通系列之:研究Debezium技术遇到的各种错误解决方法汇总: Debezium从入门到精通系列之:百篇系列文章汇总之研究Debezium技…

AI——使用pandas

文章目录 1、pandas介绍2、为什么使用pandas3、pandas的数据结构1、Series2、DataFrame3、MultiIndex 4、pandas基本数据操作1、索引操作2、赋值操作3、排序4、算术运算5、逻辑运算6、逻辑运算函数7、统计函数8、累计统计函数9、自定义运算 5、pandas读取文件和存储1、csv文件2…

2025认证杯挑战赛B题【 谣言在社交网络上的传播 】原创论文讲解(含完整python代码)

大家好呀&#xff0c;从发布赛题一直到现在&#xff0c;总算完成了认证杯数学中国数学建模网络挑战赛第一阶段B题目谣言在社交网络上的传播完整的成品论文。 本论文可以保证原创&#xff0c;保证高质量。绝不是随便引用一大堆模型和代码复制粘贴进来完全没有应用糊弄人的垃圾半…

用docker容器创建属于自己的一方小世界!容器中,盖周天之变,化吾为王~

用docker容器创建属于自己的一方小世界&#xff01;容器中&#xff0c;盖周天之变&#xff0c;化吾为王~ 分别查看用户id和组id。 命令&#xff1a; 1、id -u 2、id -g 创建并运行容器 docker run -d -p 31404:22 -v /home/liub:/home -v /data:/app/data --user 1004:1004 --…

vue拓扑图组件

vue拓扑图组件 介绍技术栈功能特性快速开始安装依赖开发调试构建部署 使用示例演示截图组件源码 介绍 一个基于 Vue3 的拓扑图组件&#xff0c;具有以下特点&#xff1a; 1.基于 vue-flow 实现&#xff0c;提供流畅的拓扑图展示体验 2.支持传入 JSON 对象自动生成拓扑结构 3.自…

Linux服务器网卡深度解析:从ifconfig输出到生产环境性能调优实战

Linux服务器网卡深度解析&#xff1a;从ifconfig输出到生产环境性能调优实战 Linux服务器网卡深度解析&#xff1a;从ifconfig输出到生产环境性能调优实战一、背景二、生产环境的服务器部署情况三、拆解一个真实的 ifconfig 输出1、先看 MAC 地址2、再看设备的 interrupt 和 me…

《嵌套调用与链式访问:C语言中的函数调用技巧》

&#x1f680;个人主页&#xff1a;BabyZZの秘密日记 &#x1f4d6;收入专栏&#xff1a;C语言 &#x1f30d;文章目入 一、嵌套调用&#xff08;一&#xff09;定义&#xff08;二&#xff09;实现方式&#xff08;三&#xff09;优点&#xff08;四&#xff09;缺点 二、链式…

Python-控制语句

控制语句 控制语句和逻辑思维 控制语句:把语句组合成能完成一定功能的小逻辑模块分类:顺序、选择、循环“顺序结构”:代表“先执行a,再执行b”的逻辑“条件判断结构”:代表“如果…,则…”的逻辑“循环结构”:代表“如果…则重复执行…”的逻辑条件判断结构 选择结构通…

教程:在Typora中显示拼音——附处理工具

原因 因为自己普通话不标准&#xff0c;希望可以制作适合自己的带拼音的文档&#xff0c;可以把平常看到的内容、说过的话作为练习普通话的材料。 在市面上&#xff0c;带拼音的材料、书籍并不多&#xff0c;而且有可能是一些比较生僻的内容。所以希望可以自己制作这样的材料…

OpenCV 图形API(30)图像滤波-----腐蚀操作函数erode()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 使用特定的结构元素腐蚀图像。 cv::gapi::erode 是 OpenCV 的 G-API 模块中用于执行图像腐蚀操作的函数。腐蚀是一种基本的形态学操作&#xff…

特殊定制版,太给力了!

今天给大家分享一款超棒的免费录屏软件&#xff0c;真的是录屏的好帮手&#xff01; 这款软件功能可以录制 MP4、AVI、WMV 格式的标清、高清、原画视频&#xff0c;满足你各种需求。 云豹录屏大师 多功能录屏神器 它的界面特别简洁&#xff0c;上手超快&#xff0c;用起来很顺…