给定一个非负整数 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这个数字被使用过了,当然这种效率很差,但也是一种思路。