forked from DengWangBao/Leetcode-Java
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSplitArrayLargestSum.java
82 lines (77 loc) · 3.05 KB
/
SplitArrayLargestSum.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
/**
* 将数组切成m段,求所有段中最大值的最小值
* 我们倒过来思考,假设这个值为k,则意味着所有段的和都不会超过k,如果我们能确保这种情况下数组能被切成m段,则表明k是候选的值。那怎样确保k是最小值呢?
* 简单的办法是让k从最小开始,依次递增,k最小是多少呢,如果不考虑具体的m,显然是数组中的最大值,而k的最大值是整个数组的和。现在我们考虑m,如果k设置的太小,
* 则为了保证所有段的和都小于k,势必段要缩短长度以降低sum,而这样的话可能段的个数会超过m。如果k设置的太大,则势必段的长度要增加,而这样的话可能段的个数会达不到m,
* 所以k和m是线性关系,为了找到最小的那个k,我们可以用二分查找法。
*/
public class SplitArrayLargestSum {
/**
* 这里二分查找时关键要注意的是只剩两个数时的情况,即r-l=1时,
* 假设最终结果是l,则mid为l,isTargetBig返回true,所以返回l
* 如果最终结果是r,则mid为l,isTargetBig返回false,所以l=mid+1=r,返回r
*/
public int splitArray(int[] nums, int m) {
int max = 0, sum = 0;
for (int i = 0; i < nums.length; i++) {
max = Math.max(max, nums[i]);
sum += nums[i];
}
int l = max, r = sum;
while (l <= r) {
int mid = l + ((r - l) >>> 1);
if (isTargetBig(mid, nums, m)) {
r = mid - 1;
} else {
l = mid + 1;
}
}
return l;
}
/**
* 如果target小了则返回false,正常或者大了则返回true
*/
private boolean isTargetBig(int target, int[] nums, int m) {
int sum = 0, count = 1;
for (int i = 0; i < nums.length; i++) {
sum += nums[i];
if (sum > target) {
sum = nums[i];
if (++count > m) {
return false;
}
}
}
return true;
}
/** 这种写法好理解一些,注意count==m时,也要缩小右边界,因为我们要找到尽可能小的值,假如mid恰好就是最小的值,之后left会加回来的
public int splitArray(int[] nums, int m) {
int max = 0, sum = 0;
for (int n : nums) {
max = Math.max(max, n);
sum += n;
}
int left = max, right = sum;
while (left <= right) {
int mid = left + (right - left) / 2;
int count = getCount(nums, mid);
if (count > m) {
left = mid + 1;
} else if (count <= m) {
right = mid - 1;
}
}
return left;
}
private int getCount(int[] nums, int k) {
int sum = 0, count = 1;
for (int i = 0; i < nums.length; i++) {
sum += nums[i];
if (sum > k) {
count++;
sum = nums[i];
}
}
return count;
}*/
}