给定一个非负整数 n,计算各位数字都不同的数字 x 的个数,其中 0 ≤ x < 10^n 。

示例:

输入: 2
输出: 91
解释: 答案应为除去 11,22,33,44,55,66,77,88,99 外,在 [0,100) 区间内的所有数字。

答案:

 1public int countNumbersWithUniqueDigits1(int n) {
2    if (n == 0)
3        return 1;
4    int res = 10;
5    int uniqueDigits = 9;
6    int availableNumber = 9;
7    while (n-- > 1 && availableNumber > 0) {
8        uniqueDigits = uniqueDigits * availableNumber;
9        res += uniqueDigits;
10        availableNumber--;
11    }
12    return res;
13}

解析:

这题没什么难度,只要上过高中,学过排列组合的估计都能看懂,我简单介绍一下,当n=1的时候也就是从0-9有多少个数各位数字都不同,很明显是10,当n等于2的时候也就是从10-99有多少个数各位数字都不同,根据排序组合先从1-9中选择一个数字x作为十位数(十位数不能是0),然后再从0-9中选择一个数字y作为个位数,组成一个新的数,其中x!=y,选择x有9种方式,选择y也有9种方式,所以有81种,再加上前面的10种,总共91种。同理当n=3的时候从100-999中选择有9*9*8种,当n=4的时候从1000-9999中选择有9*9*8*7种,所以规律很好发现,明白了这点,代码就容易理解多了,我们还可以再来修改一下

 1public int countNumbersWithUniqueDigits(int n) {
2    int res[] = new int[n + 1];
3    res[0] = 1;
4    int sum = 1;
5    int k = 9;
6    for (int i = 1; i <= n && k > 0; i++) {
7        if (i == 1)
8            res[i] += res[i - 1] * 9;
9        else
10            res[i] += res[i - 1] * k--;
11        sum += res[i];
12    }
13    return sum;
14}

上面两种虽然写法有点区别,但思路都是一样的。下面再来看一种递归的方式。

 1public int countNumbersWithUniqueDigits3(int n) {
2    return doCount(n, new boolean[10], 0);
3}
4
5private int doCount(int n, boolean[] used, int d) {
6    if (d == n)
7        return 1;
8    int total = 1;
9    for (int i = (d == 0) ? 1 : 0; i <= 9; i++) {
10        if (!used[i]) {
11            used[i] = true;
12            total += doCount(n, used, d + 1);
13            used[i] = false;
14        }
15    }
16    return total;
17}

这种解法稍微有一点难度,需要有一定的算法基础知识,否则不容易看懂。他使用递归加回溯的方式,used[i]表示i这个数字被使用过了,当然这种效率很差,但也是一种思路。