forked from gouthampradhan/leetcode
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathTwentyFourGame.java
112 lines (100 loc) · 3.53 KB
/
TwentyFourGame.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
package divide_and_conquer;
import java.util.*;
/**
* Created by gouthamvidyapradhan on 18/05/2019 You have 4 cards each containing a number from 1 to
* 9. You need to judge whether they could operated through *, /, +, -, (, ) to get the value of 24.
*
* <p>Example 1: Input: [4, 1, 8, 7] Output: True Explanation: (8-4) * (7-1) = 24 Example 2: Input:
* [1, 2, 1, 2] Output: False Note: The division operator / represents real division, not integer
* division. For example, 4 / (1 - 2/3) = 12. Every operation done is between two numbers. In
* particular, we cannot use - as a unary operator. For example, with [1, 1, 1, 1] as input, the
* expression -1 - 1 - 1 - 1 is not allowed. You cannot concatenate numbers together. For example,
* if the input is [1, 2, 1, 2], we cannot write this as 12 + 12.
*
* <p>Solution O(1) Generate all permutation of a given 4 digit number and for each permutation
* split the number into two parts with left and right (at various possible split points). Now
* perform all possible operations(+, -, * and /) for each left and right and check if any of the
* operation results in 24
*/
public class TwentyFourGame {
public static void main(String[] args) {
int[] A = {4, 7, 7, 7};
System.out.println(new TwentyFourGame().judgePoint24(A));
}
class Fraction {
int n, d;
Fraction(int n, int d) {
this.n = n;
this.d = d;
}
}
public boolean judgePoint24(int[] nums) {
List<int[]> result = new ArrayList<>();
permute(0, nums, result);
for (int[] A : result) {
List<Fraction> list = generate(0, 3, A);
for (Fraction f : list) {
if ((f.d != 0) && (f.n / f.d) == 24 && (f.n % f.d) == 0) return true;
}
}
return false;
}
private void permute(int i, int[] nums, List<int[]> result) {
if (i >= nums.length) {
result.add(Arrays.copyOf(nums, 4));
} else {
for (int j = i; j < nums.length; j++) {
swap(i, j, nums);
permute(i + 1, nums, result);
swap(i, j, nums);
}
}
}
private void swap(int i, int j, int[] nums) {
int temp = nums[i];
nums[i] = nums[j];
nums[j] = temp;
}
private List<Fraction> generate(int l, int r, int[] nums) {
if (l > r) {
return new ArrayList<>();
} else if (l == r) {
return Arrays.asList(new Fraction(nums[l], 1));
} else {
List<Fraction> result = new ArrayList<>();
for (int i = l; i < r; i++) {
for (int j = l; j <= i; j++) {
List<Fraction> left = generate(l, j, nums);
List<Fraction> right = generate(j + 1, r, nums);
if (right.isEmpty()) {
result.addAll(left);
} else if (left.isEmpty()) {
result.addAll(right);
} else {
for (Fraction lF : left) {
for (Fraction rF : right) {
int n = (lF.n * rF.d + rF.n * lF.d);
int d = (lF.d * rF.d);
Fraction sum = new Fraction(n, d);
n = (lF.n * rF.d - (rF.n * lF.d));
d = (lF.d * rF.d);
Fraction diff = new Fraction(n, d);
n = (lF.n * rF.n);
d = (lF.d * rF.d);
Fraction prod = new Fraction(n, d);
n = (lF.n * rF.d);
d = (lF.d * rF.n);
Fraction div = new Fraction(n, d);
result.add(sum);
result.add(diff);
result.add(prod);
result.add(div);
}
}
}
}
}
return result;
}
}
}