package class05; /** * 区间和的个数 * <p> * https://leetcode.com/problems/count-of-range-sum/ * 给你一个整数数组 nums 以及两个整数 lower 和 upper 。求数组中,值位于范围 [lower, upper] (包含 lower 和 upper)之内的 区间和的个数。 * 区间和 S(i, j) 表示在 nums 中,位置从 i 到 j 的元素之和,包含 i 和 j (i ≤ j)。 */ public class Code01_CountOfRangeSum { public static int countRangeSum(int[] nums, int lower, int upper) { if (nums == null || nums.length == 0) { return 0; } long[] sum = new long[nums.length];//定义一个前缀和数组 sum[0] = nums[0]; for (int i = 1; i < nums.length; i++) { //前缀和数组中,arr[i]等于,原数组nums[0]-nums[i]的累加和。 //也就是原数组nums中,它(sum[i])前面所有数的累加和,再加上它自己。 sum[i] = sum[i - 1] + nums[i]; } return process(sum, 0, sum.length - 1, lower, upper); } private static int process(long[] sum, int L, int R, int lower, int upper) { //在判断(子数组1-子数组2)在不在[lower, upper]上时, //当子数组2是空数组,即一个元素也没有时,也就是子数组1就是整个数组本身时: if (L == R) { //如果是真,就返回有一个达标的数组。否则,就返回没有达标的数组。 return sum[L] >= lower && sum[L] <= upper ? 1 : 0; } int M = L + ((R - L) >> 1); int process1 = process(sum, L, M, lower, upper); int process2 = process(sum, M + 1, R, lower, upper); int merge = merge(sum, L, M, R, lower, upper); return process1 + process2 + merge; } //合并 private static int merge(long[] sum, int L, int M, int R, int lower, int upper) { int ans = 0; int windowL = L; int windowR = L; //窗口范围是:[windowL, windowR) //从右组中开始,也就是从M+1位置开始。 for (int i = M + 1; i <= R; i++) { //使用前缀和,分解原问题。 //区间和 S(i, j),相当于是:sum[j]-sum[i-1],即(sum[0]~sum[j])-(sum[0]~sum[i-1])。 //原问题:求数组中,值位于范围 [lower, upper] (包含 lower 和 upper)之内的 区间和的个数。 long min = sum[i] - upper; long max = sum[i] - lower; while (windowR <= M && sum[windowR] <= max) {//当sum[windowR]大于max时,跳出。 windowR++; } while (windowL <= M && sum[windowL] < min) {//当sum[windowL]等于min时,跳出。 windowL++; } //此时的[windowL, windowR)上的每一个数,都是达标的数字。 //因为sum是前缀和数组,不是原数组nums,所以sum中的每一个数字,就代表一个区间和。并且是从0到当前索引位置。 ans += windowR - windowL;//每次merge,累加上ans。每次merge,累加上ans。最后ans就是所有达标的对数之和。 } //经典归并排序: long[] help = new long[R - L + 1]; int i = 0; int p1 = L; int p2 = M + 1; while (p1 <= M && p2 <= R) { help[i++] = sum[p1] <= sum[p2] ? sum[p1++] : sum[p2++]; } while (p1 <= M) { help[i++] = sum[p1++]; } while (p2 <= R) { help[i++] = sum[p2++]; } for (i = 0; i < help.length; i++) { sum[L + i] = help[i]; } return ans; } public static int[] generatorRandomArray(int maxSize, int maxValue) { int[] arr = new int[(int) (Math.random() * maxSize) + 2]; for (int i = 0; i < arr.length; i++) { arr[i] = (int) (Math.random() * maxValue) - ((int) (Math.random() * maxValue)); } return arr; } public static int[] copyArray(int[] arr) { if (arr == null) { return null; } int[] res = new int[arr.length]; for (int i = 0; i < arr.length; i++) { res[i] = arr[i]; } return res; } public static void main(String[] args) { int maxSize = 100; int maxValue = 100; int testTimes = 100000; System.out.println("test start!"); for (int i = 0; i < testTimes; i++) { int lower; int upper; do { lower = (int) (Math.random() * maxValue); upper = (int) (Math.random() * maxValue); } while (lower > upper); int[] arr0 = generatorRandomArray(maxSize, maxValue); int[] arr1 = copyArray(arr0); int[] arr2 = copyArray(arr0); int res = countRangeSum(arr1, lower, upper); int test = test(arr2, lower, upper); if (res != test) { System.out.println("oops!"); break; } } System.out.println("test end!"); } //暴力解法 public static int test(int[] arr, int lower, int upper) { int count = 0; for (int i = 0; i < arr.length; i++) { for (int j = i; j < arr.length; j++) { int ans = 0; for (int m = i; m <= j; m++) { ans += arr[m]; } // System.out.print(ans + " "); if (ans >= lower && ans <= upper) { count++; } } // System.out.println(); } // System.out.println("count = " + count); return count; } } /* * 提示: * 假设0-i整体的累加和是x, * 求以i位置结尾的所有的子数组中,有多少个在[lower, upper]上, * 等同于去求,i之前的所有前缀和中,有多少个前缀和,在[x-upper, x-lower]上。 * * 小结: * 1.前缀和可以做一些事情。 * 2.将原问题转换成,另一个问题,即原数组中,值位于[x-upper, x-lower]上的区间和的个数。(x是前缀和sum中的某一个值) * 原问题:求数组中,值位于范围 [lower, upper] (包含 lower 和 upper)之内的 区间和的个数。 * 3.前缀和中,x之前,有多少个数,在[x-upper, x-lower]上。 * 4.在归并排序的过程中做这些事。 * 5.要想merge左组和右组,是O(N)的时间复杂度,必须知道p1,p2不回退的技巧。 */
标签:upper,lower,nums,int,sum,个数,数组,区间 From: https://www.cnblogs.com/TheFloorIsNotTooHot/p/16879305.html