哈希集合
请读者思考,num拆分成num1和num2,要使得num1 + num2最小,应满足两条性质:
- num1和num2位数相同,或最多差一位。
- num1和num2应按数值从小到大在num中取数。
想到统计num的位数,以实现性质1的需要;想到用哈希集合a[10],统计数字{0~9}在num中出现的次数,以实现性质2的需要。
请看更具体的性质:
- num是奇数时,num1和num2最多差一位;num是偶数时,num1和num2位数相同。
- num1先取一个数,num2再取一个数,取数时按数值从小到大在num中取数。
请看下列代码:
class Solution {
public:int splitNum(int num) {int a[10] = { 0 };int count = 0;int num1 = 0, num2 = 0;while (num) {count ++;a[num % 10] ++;num /= 10;}if (count & 1) { // 奇数count --;for (int i = 0; i < 10; i ++) {if (a[i]) {a[i] --;num1 += i;break;}}} while (count) {for (int i = 0; i < 10; i ++) {if (a[i]) {num1 *= 10;num1 += i;a[i] --;count --;break;}}for (int i = 0; i < 10; i ++) {if (a[i]) {num2 *= 10;num2 += i;a[i] --;count --;break;}}}return num1 + num2;}
};
更具体的,我们发现奇数无需区分,请看推理:
设num是奇数,num1多取了一位数(比num2多一位)。不考虑num1的首位,num1的后续排列只有两种(交替取数),且num1的排列刚好和num2的排列相反。因此{num1的排列} + {num2的排列}是一个定值(按性质1/2交替取数的前提下!!),所以奇偶数无需区分。
class Solution {
public:int splitNum(int num) {int a[10] = { 0 };int count = 0;int num1 = 0, num2 = 0;while (num) {count ++;a[num % 10] ++;num /= 10;}while (count) {for (int i = 0; i < 10; i ++) {if (a[i]) {num1 *= 10;num1 += i;a[i] --;count --;break;}}for (int i = 0; i < 10; i ++) {if (a[i]) {num2 *= 10;num2 += i;a[i] --;count --;break;}}}return num1 + num2;}
};
时间复杂度 O ( l o g n ) O(logn) O(logn) : n n n 是数字大小,按数位处理数字的时间复杂度 O ( l o g 10 n ) O(log_{10}n) O(log10n) 。忽略常数,时间复杂度 O ( l o g n ) O(logn) O(logn) 。
空间复杂度 O ( ∣ C ∣ ) O(|C|) O(∣C∣) : 哈希集合的大小 ∣ C ∣ |C| ∣C∣,空间复杂度 O ( ∣ C ∣ ) O(|C|) O(∣C∣) 。
AC
致语
- 理解思路很重要
- 读者有问题请留言,清墨看到就会回复的。