Topk算法_topn算法

Topk算法_topn算法topK算法思路1:可以采用快速选择算法,借助快排,设mid为每次划分中间结果,每次划分完之后如果mid==k,则说明序列刚刚好,第k位置和他前面的位置都是前K大的数,如果mid < k,则说明第K大的元素在后半部分,则前半部分肯定是前K大的数,只需从后半部分找k – mid大的数即可,否则如果mid > k,则说明第K大的数在前半部分,只需从前半部分找前K大的数字即可。时间复杂度:假设每次划分的mid都在中间,每层都只是对一半做划分,所以每次划分的数据量为n,n/2,n/4,n/8…一

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

topK算法

思路1:快速选择算法
可以采用快速选择算法,借助快排,设mid为每次划分中间结果,每次划分完之后如果mid==k,则说明序列刚刚好,第k位置和他前面的位置都是前K大的数,如果mid < k,则说明第K大的元素在后半部分,则前半部分肯定是前K大的数,只需从后半部分找k – mid大的数即可,否则如果mid > k,则说明第K大的数在前半部分,只需从前半部分找前K大的数字即可。
时间复杂度:假设每次划分的mid都在中间,每层都只是对一半做划分,所以每次划分的数据量为
n,n/2,n/4,n/8…一共有logn层,根据等比数列可以算出来时间复杂度为O(n)
C++代码演示

#include<bits/stdc++.h>
#define x first
#define y second
#define send string::npos
#define lowbit(x) (x&(-x))
#define left(x) x<<1
#define right(x) x<<1|1
using namespace std;
typedef long long ll;
typedef pair<int,int> PII;
typedef struct Node * pnode;
const int N = 2e5 + 10;
const int M = 3 * N;
const int INF = 0x3f3f3f3f;
const int Mod = 1e9;
int partion(int l,int r,int a[]){ 
   
    int x = a[l];
    while(l < r){ 
   
        while(l < r && a[r] < x)r --;
        if(l < r)a[l ++] = a[r];
        while(l < r && a[l] > x)l ++;
        if(l < r)a[r --] = a[l];
    }
    a[l] = x;
    return l;
}
int quickSort(int l,int r,int a[],int k){ 
   
    if(l < r){ 
   
        int mid = partion(l,r,a);		//划分结果
        int ls = mid - l + 1;   //[l,mid]一共有多少数
        if(ls == k)return;      //如果划分的mid恰好就是第K大的数
        if(ls < k)quickSort(mid + 1,r,a,k - ls);   //如果第K大的数在后半部分
        else quickSort(l,mid - 1,a,k);  //如果第K大的数在前半部分
    }
}
int main(){ 
   
    int a[] = { 
   3,2,3,1,2,4,5,5,6};
    int k = 4;
    quickSort(0,8,a,k);
    for(int i = 0;i < k;i ++)
        cout<<a[i]<<" ";
    return 0;
}

思路2:堆排序
可以建立一个大小为K的小顶堆,每次把当前数和小顶堆的堆顶作比较,如果当前数字大,则替换掉堆顶,重新调整堆。
时间复杂度:调整堆为O(logK),所以总的时间复杂度为nlog(K)

C++代码

#include<bits/stdc++.h>
#define x first
#define y second
#define send string::npos
#define lowbit(x) (x&(-x))
#define left(x) x<<1
#define right(x) x<<1|1
using namespace std;
typedef long long ll;
typedef pair<int,int> PII;
typedef struct Node * pnode;
const int N = 2e5 + 10;
const int M = 3 * N;
const int INF = 0x3f3f3f3f;
const int Mod = 1e9;
priority_queue<int,vector<int>,greater<int> >pq;  //默认是大顶堆
int main(){ 
   
    int k = 3;
    int a[] = { 
   3,1,5,2,4,3};
    for(int i = 0;i < 3;i ++)pq.push(a[i]);
    for(int i = 3;i < 6;i ++){ 
   
        if(pq.top() < a[i]){ 
   
            pq.pop();
            pq.push(a[i]);
        }
    }
    while(!pq.empty()){ 
   
        cout<<pq.top()<<endl;
        pq.pop();
    }
    return 0;
}

对比
堆方案只需要开辟K个数组空间即可,而且不需要变动原数组,而随机选择需要改原数组,如果不改动的话就需要创建一个O(n)的数组


leetcode例题
原题链接
数组中的第K个最大元素
在未排序的数组中找到第 k 个最大的元素。请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素。

示例 1:

输入: [3,2,1,5,6,4] 和 k = 2
输出: 5
示例 2:

输入: [3,2,3,1,2,4,5,5,6] 和 k = 4
输出: 4
说明:

你可以假设 k 总是有效的,且 1 ≤ k ≤ 数组的长度。

题解
1.快速选择版本

class Solution { 
   
public:
    int partion(int l,int r,vector<int>&nums){ 
   
        int x = nums[l];
        while(l < r){ 
   
            while(l < r && nums[r] < x)r --;
            if(l < r)nums[l ++] = nums[r];
            while(l < r && nums[l] > x)l ++;
            if(l < r)nums[r --] = nums[l];
        }
        nums[l] = x;
        return l;
    }
    void quickSort(int l,int r,vector<int>&nums,int k){ 
   
        if(l < r){ 
   
            int mid = partion(l,r,nums);
            int lnum = mid - l + 1;     //[l,mid]一共有多少个数
            if(k == lnum)return;
            if(k < lnum)quickSort(l,mid - 1,nums,k);
            else quickSort(mid + 1,r,nums,(k - lnum));
        }
    }
    int findKthLargest(vector<int>& nums, int k) { 
   
        quickSort(0,nums.size() - 1,nums,k);
        return nums[k - 1];
    }
};

2.优先队列版本

class Solution { 
   
public:
    int findKthLargest(vector<int>& nums, int k) { 
   
        priority_queue<int,vector<int>,greater<int> >pq;
        for(int i = 0;i < k;i ++)pq.push(nums[i]);
        for(int i = k;i < nums.size();i ++){ 
   
            if(pq.top() < nums[i]){ 
   
                pq.pop();
                pq.push(nums[i]);
            }
        }
        return pq.top();
    }
};
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/168776.html原文链接:https://javaforall.net

(0)
上一篇 2022年8月8日 下午1:00
下一篇 2022年8月8日 下午1:00


相关推荐

  • 深入学习USB(6)USB Type-C接口定义概念解析

    深入学习USB(6)USB Type-C接口定义概念解析一 Usbtypec 接口定义介绍 USBType C 接口总计有 24 个针脚 可以正反插且传输速度快 接口没有方向性 让用户在使用中避免出现插错的情况 一般简称有 typec type c 等这些指的都是同一个产品 而 USB3 0 接口通常是 9 到 11 个 USB2 0 只有 4 个针脚 针脚的增多并没有导致 Type C 接口体积变大 实际上它还缩小了体积 相对标准口来说 满足了移动设备的需求 二 USB3 1type c 接口特性特性 1 全新接口设计 尺寸约 8 4mmx2 6mm 接口纤薄特性 2 功率输出能

    2026年3月17日
    2
  • 01-Springboot优点&缺点

    01-Springboot优点&缺点springboot 是在 ssm 框架传统开发基础上进化过来 相交于传统的开发方式 尤其优点 有其缺点 但是总的来看 优点远远大于缺点 并且 boot 已经成为开发的基础和趋势 非常值得 并且也是必须掌握的技术 是基础技术 要鬼瓜烂熟 1 Springboot 的优点 starter 想起之前开发 ssm 框架的项目 最害怕或者最让人不爽的莫过于对依赖的管理 现在有了 boot 后 就有了村长 县长 引入村长和县长后 就直接和他们说事 具体的依赖他们可以全权代表了 编码更简单了 boot 采用了 config 的方式进行配置

    2026年3月18日
    2
  • js中bind的用法_bind into

    js中bind的用法_bind intoJS中的bind的实现以及使用

    2022年4月21日
    80
  • 数组和链表的区别,各有何优缺点

    数组和链表的区别,各有何优缺点链表与数组的区别(1)数组的元素个数是固定的,而组成链表的结点个数可按需要增减;(2)数组元素的存诸单元在数组定义时分配,链表结点的存储单元在程序执行时动态向系统申请;(3)数组中的元素顺序关系由元素在数组中的位置(即下标)确定,链表中的结点顺序关系由结点所包含的指针来体现。(4)对于不是固定长度的列表,用可能最大长度的数组来描述,会浪费许多内存空间。(5)对于元素的插人、删除操作非常频繁的列表处理场合,用数组表示是不适宜的。若用链表实现,会使程序结构清晰,处理的方法也较为简便。数组的优点随机

    2022年6月17日
    91
  • 幅度调制(线性调制)原理

    幅度调制(线性调制)原理之前在书上第 5 页我们学过模拟通信系统模型和数字通信系统模型里面都有调制器今天我们来具体学一下什么是调制调制 把货物搭载到飞机的某个仓位上利用专业术语表述就是把消息搭载到载波的某个参数上载波就是高频周期性振荡信号解调就是从已调波恢复出原信号调制也分广义调制和狭义调制 广义调制就是我们学的第五页那一个很长的图狭义调制 把消息搭载的载波的某个参数上 相对广义过程很长 直接牵扯了很多与调制

    2026年3月19日
    2
  • Cursor+Claude全流程建站指南:AI驱动开发实战教程

    Cursor+Claude全流程建站指南:AI驱动开发实战教程

    2026年3月15日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号