LeetCode 493 Reverse Pairs / 315

2018-10-22  本文已影响0人  manayuan

Similarity

Three Implementation

315 Count of Smaller Numbers After Self

You are given an integer array nums and you have to return a new counts array. The counts array has the property where counts[i] is the number of smaller elements to the right of nums[i].

Example:
Input: [5,2,6,1]
Output: [2,1,1,0]
Explanation:
To the right of 5 there are 2 smaller elements (2 and 1).
To the right of 2 there is only 1 smaller element (1).
To the right of 6 there is 1 smaller element (1).
To the right of 1 there is 0 smaller element.

BST Solution

public class Solution {
    class Node {
        Node left, right;
        int val, sum, dup = 1;
        public Node(int v, int s) {
            val = v;
            sum = s;
        }
    }
    public List<Integer> countSmaller(int[] nums) {
        Integer[] ans = new Integer[nums.length];
        Node root = null;
        for (int i = nums.length - 1; i >= 0; i--) {
            root = insert(nums[i], root, ans, i, 0);
        }
        return Arrays.asList(ans);
    }
    private Node insert(int num, Node node, Integer[] ans, int i, int preSum) {
        if (node == null) {
            node = new Node(num, 0);
            ans[i] = preSum;
        } else if (node.val == num) {
            node.dup++;
            ans[i] = preSum + node.sum;
        } else if (node.val > num) {
            node.sum++;
            node.left = insert(num, node.left, ans, i, preSum);
        } else {
            node.right = insert(num, node.right, ans, i, preSum + node.dup + node.sum);
        }
        return node;
    }
}

MergeSort Solution

public class Solution {
    public List<Integer> countSmaller(int[] nums) {
        int[] count = new int[nums.length];
        int[] indexes = new int[nums.length];
        for (int i=0; i<indexes.length; i++) {
            indexes[i] = i;
        }
        mergeSort(nums, count, indexes, 0, nums.length - 1);
        List<Integer> res = new ArrayList<Integer>();
        for (int i=0; i<nums.length; i++) {
            res.add(count[i]);
        }
        return res;
    }
    public void mergeSort(int[] nums, int[] count, int[] indexes, int start, int end) {
        if (end <= start) return;
        int mid = (start + end) / 2;
        mergeSort(nums, count, indexes, start, mid);
        mergeSort(nums, count, indexes, mid + 1, end);
        merge(nums, count, indexes, start, end);
    }
    public void merge(int[] nums, int[] count, int[] indexes, int start, int end) {
        int mid = (start + end) / 2;
        int left_index = start;
        int right_index = mid+1;
        int rightcount = 0;     
        int[] new_indexes = new int[end - start + 1];

        int sort_index = 0;
        while(left_index <= mid && right_index <= end){
            if(nums[indexes[right_index]] < nums[indexes[left_index]]){
                new_indexes[sort_index] = indexes[right_index];
                rightcount++;
                right_index++;
            }else{
                new_indexes[sort_index] = indexes[left_index];
                count[indexes[left_index]] += rightcount;
                left_index++;
            }
            sort_index++;
        }
        while(left_index <= mid){
            new_indexes[sort_index] = indexes[left_index];
            count[indexes[left_index]] += rightcount;
            left_index++;
            sort_index++;
        }
        while(right_index <= end){
            new_indexes[sort_index++] = indexes[right_index++];
        }
        for(int i = start; i <= end; i++){
            indexes[i] = new_indexes[i - start];
        }
    }
}

BIT Solution

public class Solution {
    public List<Integer> countSmaller(int[] nums) {
        List<Integer> res = new ArrayList<Integer>();
        if (nums == null || nums.length == 0) return res;
        int min = Integer.MAX_VALUE;
        int max = Integer.MIN_VALUE;
        for (int num : nums) {
            min = Math.min(min, num);
        }
        int[] nums2 = new int[nums.length];
        for (int i=0; i<nums.length; i++) {
            nums2[i] = nums[i] - min + 1;
            max = Math.max(max, nums2[i]);
        }
        int[] tree = new int[max + 1];
        for (int i=nums2.length-1; i>=0; i--) {
            res.add(0, query(nums2[i] - 1, tree));
            update(nums2[i], tree);
        }
        return res;
    }
    private int query(int i, int[] tree) {
        int num = 0;
        while (i > 0) {
            num += tree[i];
            i -= i & (-i);
        }
        return num;
    }
    private void update(int i, int[] tree) {
        while (i < tree.length) {
            tree[i] ++;
            i += i & (-i);
        }
    }
}

493 Reverse Pairs

BST Solution

class Solution {
    public int reversePairs(int[] nums) {
        Node root = null;
        int cnt = 0;
        for (int i=nums.length-1; i>=0; i--) {
            cnt = search(cnt, root, nums[i] / 2.0);
            root = buildTree(nums[i], root);
        }
        return cnt;
    }
    private Node buildTree(int val, Node node) {
        if (node == null) return new Node(val);
        if (node.val == val) {
            node.dup ++;
        }
        else if (node.val > val) {
            node.less ++;
            node.left = buildTree(val, node.left);
        }
        else {
            node.right = buildTree(val, node.right);
        }
        return node;
    }
    private int search(int cnt, Node node, double target) {
        if (node == null) return cnt;
        if (target < node.val) {
            return search(cnt, node.left, target);
        }
        else if (target == node.val) {
            return cnt + node.less;
        }
        else {
            return search(cnt + node.less + node.dup, node.right, target);
        }
    }
}
class Node {
    int val, less, dup;
    Node left, right;
    public Node(int _val) {
        val = _val;
        dup = 1;
    }
}

MergeSort Solution

class Solution {
    public int reversePairs(int[] nums) {
        return mergeSort(nums, 0, nums.length - 1);
    }
    private int mergeSort(int[] nums, int start, int end) {
        if (start >= end) return 0;
        int mid = start + (end - start) / 2;
        int cnt = mergeSort(nums, start, mid) + mergeSort(nums, mid+1, end);
        for (int i=start, j=mid+1; i<=mid; i++) {
            while (j <= end && nums[i] / 2.0 > nums[j]) j ++;
            cnt += j - (mid + 1);
        }
        Arrays.sort(nums, start, end + 1);
        return cnt;
    }
}

BIT Solution

class Solution {
    public int reversePairs(int[] nums) {
        int n = nums.length;
        
        // why long? 2 * nums[i] could overflow
        // why this array? need its indexes to build bit
        long[] nums2 = new long[n];
        for (int i = 0; i < n; i++) {
            nums2[i] = (long)nums[i];
        }
        Arrays.sort(nums2);
        
        int[] bit = new int[n + 1];
        
        int count = 0;
        
        for (int i = 0; i < n; i++) {
            int curr = nums[i];
            
            int idx1 = Arrays.binarySearch(nums2, 2l * curr + 1);
            count += query(bit, (idx1 < 0 ? -(idx1 + 1) : idx1) + 1);

            int idx2 = Arrays.binarySearch(nums2, curr);
            insert(bit, (idx2 < 0 ? -(idx2 + 1) : idx2) + 1, 1);
        }
        
        return count;    
    }
    
    private int query(int[] bit, int x) {
        int res = 0;
        
        for (int i = x; i < bit.length; i += i & -i) {
            res += bit[i];
        }
        
        return res;
    }
    
    private void insert(int[] bit, int x, int val) {
        for (int i = x; i > 0; i -= i & -i) {
            bit[i] += val;
        }
    }
}

Reference

General principles behind problems similar to "Reverse Pairs"

上一篇 下一篇

猜你喜欢

热点阅读