Problem
You are given an integer n
. We reorder the digits in any order (including the original order) such that the leading digit is not zero.
Return true
if and only if we can do this so that the resulting number is a power of two.
Examples
Example 1:
Input: n = 1
Output: true
Example 2:
Input: n = 10
Output: false
Constraints
1 <= n <= 109
Solution
Method 1 - Generate Powers of Two and Compare Anagrams
We know power of 2:
1 -> 1
2 -> 10
4 -> 100
8 -> 1000
16 -> 10000
64 -> 100000
128 -> 1000000
256 -> 10000000
...
Consider the input number 46. The answer should be true because 64 is a power of 2. Generating all permutations of 46 is time-consuming.
Instead, we can generate and sort the digits of powers of 2, for example, 64 remains 64, and 1024 becomes 4210 when sorted. When given an input number like 46, we can sort its digits to get 64, which matches a power of 2.
To generate powers of 2, we can use the left shift operator, continuing until the value exceeds the constraint of 10^9.
This problem in a way is similar to Group Anagrams.
Code
Java
public boolean reorderedPowerOf2(int N) {
String s1 = transform(N);
for (int i = 0; i < 31; i++) {
int powerOfTwo = (int)(1 << i);
String s2 = transform(powerOfTwo);
if (s1.equals(s2)) {
return true;
}
}
return false;
}
private String transform(int num) {
char[] a1 = String.valueOf(num).toCharArray();
Arrays.sort(a1);
return new String(a1);
}
Complexity
- Time:
O(m log m)
, wherem
is number of digits- Transform method takes
O(m)
- And we call transform method
log m
time
- Transform method takes
- Space:
O(m)
Method 2 - Generate Power of Twos and Compare Digits
Code
Java
public boolean reorderedPowerOf2(int n) {
char[] number = sortedDigits(n);
for (int i = 0; i < 30; ++i) {
char[] powerOfTwo = sortedDigits(1<< i);
if (Arrays.equals(number, powerOfTwo)) {
return true;
}
}
return false;
}
private char[] sortedDigits(int n) {
char[] digits = String.valueOf(n).toCharArray();
Arrays.sort(digits);
return digits;
}
Method 3 - Generate Power of Twos and Compare Digit Frequency
Code
Java
public boolean reorderedPowerOf2(int N) {
int[] inputDigitFrequency = getDigitFrequency(N);
for (int i = 0; i<31; i++) {
int twoToPowerOfI = (int) Math.pow(2, i);
int[] powerOf2FreqCount = getDigitFrequency(twoToPowerOfI);
if (compareArray(inputDigitFrequency, powerOf2FreqCount)) {
return true;
}
}
return false;
}
private int[] getDigitFrequency(int n) {
int[] digitFreq = new int[10];
while (n > 0) {
digitFreq[n % 10]++;
n = n / 10;
}
return digitFreq;
}
private boolean compareArray(int[] freqCount1, int[] freqCount2) {
for (int i = 0; i<10; i++) {
if (freqCount1[i] != freqCount2[i]) {
return false;
}
}
return true;
}
Complexity
- Time:
O(NNNXXXNNN)
- Space:
O(NNNXXX)