LeetCode 357. Count Numbers with Unique Digits

本文探讨了如何计算[0,10^n)范围内每位数字都不同的整数个数,通过动态规划的方法解决了这一问题,并提供了详细的解题思路及C++实现代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

357. Count Numbers with Unique Digits

Given a non-negative integer n, count all numbers with unique digits, x, where 0x<10n.

Example:
Given n = 2, return 91. (The answer should be the total numbers in the range of 0 ≤ x < 100, excluding [11,22,33,44,55,66,77,88,99])

题目内容:
题目一个非负整数n,让我们算出[0,10n)之间每一位数字都不相同的整数个数。

解题思路:
因为对于一个n(n2),我们可以通过分别计算[0,10n1)[10n1,10n)的之间每一位数字都不相同的整数个数,然后相加获得。如果令result[n]是题目要求的结果,那么其实也也算是一个动态规划的问题,因为result[n]可以通过result[n1]来算出。
那么,对于一个n(n2),假如我们已经知道result[n1],如何算出[10n1,10n)每一位数字都不相同的整数个数。如下图,对于[10n1,10n)区间里面的数,假如我们把这个数分成2个部分,第一部分是最高位,第二部分是剩下的位。
这里写图片描述
那么我们现在要做的就是往这n个格子里面分别放入不同的数字,其中第n位不为0。可以分成两种情况。

  1. n-1位里面没有0。那么n-1位这部分共有An19种可能情况,第n位就有9(n1)种可能。组合起来总共An199(n1)种可能。
  2. n-1位里面有0。那么n-1位这部分共有An29(n1)种可能情况,第n位就有10(n1)种可能。组合起来总共An29(n1)(10(n1))种可能。
    所以,对于[10n1,10n)这个区间,每一位数字都不相同的数字个数有
    An199(n1)+An29(n1)(10(n1))

那么result的状态转移方程就是

result[n]=10n==1

result[n]=result[n1]+An199(n1)+An29(n1)(10(n1)),n2

代码:

class Solution {
public:
    int countNumbersWithUniqueDigits(int n) {
        if (n == 0) return 1;
        else if (n == 1) return 10;
        int result = 10;
        for (int i = 2; i <= n; i++) {
            result += (i - 1) * A(9, i - 2) * (11 - i) + A(9, i - 1) * (10 - i);
        }
        return result;
    }

    int A(int a, int b) {
        if (b == 0) return 1;
        int r = 1;
        for (int i = 0; i < b; i++) {
            r *= (a - i);
        }
        return r;
    }

};
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值