算法-TopK问题

数组中的第K个最大元素(经典题)

添加链接描述

大小为K的小根堆

建一个大小为K的小根堆,那么最后的堆顶就是第K大的值

class Solution {
    
    
    public int findKthLargest(int[] nums, int k) {
    
    
        int[]heap=new int[k];
        for (int i=0;i<nums.length;i++){
    
    
            if (i<k){
    
    
                heap[i]=nums[i];
                if (i==k-1){
    
    
                    // 建堆:从第一个非叶子节点到堆顶节点进行调整,调整完就是一个初始的小根堆
                    for (int j=k/2-1;j>=0;j--){
    
    
                        heapSort(heap,j);
                    }
                }
            }else {
    
    
               if (nums[i]>heap[0]){
    
    
                   //替换堆顶并重新调整
                   heap[0]=nums[i];
                   heapSort(heap, 0);
               }
            }
        }
        return heap[0];
    }

    private void heapSort(int[] heap, int j) {
    
    
        int k=j*2+1;
        while (k<heap.length){
    
    
            if (k+1<heap.length && heap[k+1]<heap[k]){
    
    
                k++;
            }
            if (heap[j]>heap[k]){
    
    
                swap(heap,k,j);
                j=k;
                k=j*2+1;
            }else {
    
    
                break;
            }
        }
    }

    private void swap(int[] heap, int k, int j) {
    
    
        int temp=heap[k];
        heap[k]=heap[j];
        heap[j]=temp;
    }
}

快排

在经典快排基础上有一个优化,

class Solution {
    
    
    public int findKthLargest(int[] nums, int k) {
    
    
        return quickSelect(nums, 0, nums.length - 1, nums.length - k);
    }

    public int quickSelect(int[] a, int l, int r, int index) {
    
    
       if (left>=right){
    
    
            return nums[left];
        }
        int q = partition(a, l, r);
        if (q == index) {
    
    
            return a[q];
        } else {
    
    
        //优化
            return q < index ? quickSelect(a, q + 1, r, index) : quickSelect(a, l, q - 1, index);
        }
    }

    
    public int partition(int[] a, int l, int r) {
    
    
    // 随机找一个值作为povit并移到最后一个位置
        Random random = new Random();
        int i = random.nextInt(r - l + 1) + l;
        swap(a, i, r);
      // 快排核心 
        int povit = a[r];
        i = l;
        for (int j = l; j < r; ++j) {
    
    
            if (a[j] <= povit ) {
    
    
                swap(a, i++, j);
            }
        }
        swap(a, i , r);
        return i; //此时i就是已经定好位置的元素下标,左边的值都比a[i] 小,右边的值都比a[i]大
    }

    public void swap(int[] a, int i, int j) {
    
    
        int temp = a[i];
        a[i] = a[j];
        a[j] = temp;
    }
}

前 K 个高频元素

前 K 个高频元素
方法和上题一模一样,只是多用一个hashMap记录出现的次数

class Solution {
    
    
    //<元素,元素出现的个数>
    Map<Integer , Integer> map = new HashMap<>();

    public int[] topKFrequent(int[] nums, int k) {
    
    
       
        for(int i : nums){
    
    
            map.put(i , map.getOrDefault(i,0) + 1);
        }
        
        //得到数组中的所有元素
        Iterator it = map.keySet().iterator();
        
        // 定义heap数组,下标从1开始
        int[] heap = new int[k+1];
        int i = 1;
        while(it.hasNext()){
    
    
            // 先初始化一个大小为k的堆,当堆中元素个数为k时,建立小根堆
            if( i <= k){
    
    
                heap[i] = (Integer)it.next();
                
                if(i == k){
    
    
                    for(int j = k/2 ; j >= 1 ; j--){
    
    
                        heapSort(heap,j);
                    }
                }
                i++;
            }else{
    
       //小根堆建好后 ,对于每个新遍历的元素与堆顶比较,如果比堆顶大,替换堆顶,重新维持小根堆
                int key = (Integer)it.next();
                if(map.get(key) > map.get(heap[1])){
    
    
                   heap[1] = key;
                   heapSort(heap , 1);                
                } 
            }

        }
        
        // heap数组下标0没有使用,需要去掉
        int[] ans=new int[k];
        for(int j=1;j<=k;j++){
    
    
            ans[j-1]=heap[j];
        }
        return ans;
    }
    //维持小根堆
    //注意,堆中元素比较是比较它们出现的次数,即该元素在map中对应的value
    public void heapSort(int[] heap , int i){
    
    
        int j=i*2; // 0*2=0 所有下标要从1开始
        while (j<heap.length){
    
                     // j+1比j小的话,就要把j+1换上去。
            if(j+1<heap.length && map.get(heap[j+1])<map.get(heap[j])){
    
    
                j++;
            }
            if(map.get(heap[i])>map.get(heap[j])){
    
    
                int temp = heap[i];
                heap[i]=heap[j];
                heap[j]=temp;
                i=j;
                j=j*2;
            }else {
    
    
                break;
            }
        }
    }
}

优先队列

猜你喜欢

转载自blog.csdn.net/qq_40337086/article/details/115565939
今日推荐