对于曾经,假设要我求第k小元素。或者是求前k大元素,我可能会将元素先排序,然后就直接求出来了,可是如今有了更好的思路。

一.线性时间内求第k小元素

这个算法又是一个基于分治思想的算法。

其详细的分治思路例如以下:

1.分解:将A[p,r]分解成A[p,q-1]和A[q+1,r]两部分。使得A[p,q-1]都小于A[q],A[q+1,r]都不小于A[q];

2.求解:假设A[q]恰好是第k小元素直接返回,假设第k小元素落在前半区间就到A[p,q-1]递归查找。否则到A[q+1,r]中递归查找。

3.合并:这个问题不须要合并。

其相应的代码例如以下:

  1. int RandomziedSelect(int *a,int p,int r,int k)
  2. {
  3. if(p==r)///假设当前区间仅仅剩一个元素,那么这个元素一定就是我们要求的
  4. return a[p];
  5. int q=RandomParatition(a,p,r); ///随机划分函数
  6. int x=q-p+1;///求出a[p,q]之间的长度
  7. if(x==k) ///a[q]恰好是第k小元素
  8. return a[q];
  9. if(x>k) ///x小于k说明第k小元素在a[p,q-1]之间
  10. return RandomziedSelect(a,p,q-1,k);
  11. else ///x大于k说明第k小元素在a[q+1,r]之间,并且是这个区间的第k-x小元素
  12. return RandomziedSelect(a,q+1,r,k-x);
  13. }

事实上这个过程非常相似于快排,可是为什么快排的时间复杂度是O(nlgn),而这个算法的时间复杂度仅仅有O(n)?基本的原因在于这个算法每次仅仅要处理分解以后一半的区间,而不像快排那样两边都要处理。

当然这仅仅是一个简单的分析,更详细数学分析在这里就不说了。事实上我们也能够利用堆的性质来求出第k小元素,仅仅要我们建立一个最小堆后然后再调整k-1次即可了,这样时间复杂度是O(n)+O((k-1)lgn)。

以下给出一份完整的代码:

  1. #include<iostream>
  2. #include<cstring>
  3. #include<cstdio>
  4. #include<cmath>
  5. #include<algorithm>
  6. #include<ctime>
  7. #include<fstream>
  8. using namespace std;
  9. int Paratition(int *a,int p,int r)
  10. {
  11. int key=a[r];
  12. int i=p-1;
  13. for(int j=p;j<r;j++)
  14. if(a[j]<key)
  15. {
  16. i++;
  17. swap(a[i],a[j]);
  18. }
  19. swap(a[i+1],a[r]);
  20. return i+1;
  21. }
  22. int RandomParatition(int *a,int p,int r)
  23. {
  24. int x=rand()%(r-p+1)+p;///产生[p,r]之间的随机数
  25. swap(a[x],a[r]); ///交换a[x]和a[r]的值,事实上就是将a[x]作为划分的关键值
  26. return Paratition(a,p,r);
  27. }
  28. int RandomziedSelect(int *a,int p,int r,int k)
  29. {
  30. if(p==r)///假设当前区间仅仅剩一个元素,那么这个元素一定就是我们要求的
  31. return a[p];
  32. int q=RandomParatition(a,p,r); ///随机划分函数
  33. int x=q-p+1;///求出a[p,q]之间的长度
  34. if(x==k) ///a[q]恰好是第k小元素
  35. return a[q];
  36. if(x>k) ///x小于k说明第k小元素在a[p,q-1]之间
  37. return RandomziedSelect(a,p,q-1,k);
  38. else ///x大于k说明第k小元素在a[q+1,r]之间,并且是这个区间的第k-x小元素
  39. return RandomziedSelect(a,q+1,r,k-x);
  40. }
  41. int main()
  42. {
  43. int b[100];
  44. ifstream fin("lkl.txt");
  45. int n,k;
  46. //cout<<"请输入n,k: ";
  47. fin>>n>>k;
  48. //cout<<"请输入"<<n<<"个元素: "<<endl;
  49. for(int i=1;i<=n;i++)
  50. fin>>b[i];
  51. int ans=RandomziedSelect(b,1,n,k);
  52. sort(b+1,b+n+1);
  53. for(int i=1;i<=n;i++)
  54. cout<<b[i]<<" ";
  55. cout<<endl;
  56. cout<<"第"<<k<<"小元素为: "<<ans<<endl;
  57. return 0;
  58. }

二.利用堆求前k大元素

这个算法的思想比較简单: 假设我们要求n个元素中前k大的元素。我们就先将这n个元素中的前k个元素建立一个最小堆,然后从k+1。

。。

n依次推断。假设某个元素大于堆中最小的元素,我们就将其替代堆中的最小元素,并且调整一下堆。

这样将全部元素都检查完了之后,堆中的k个元素也就是这n个元素中的前k大元素了。时间复杂度O(k)+O((n-k)lgk)。

代码例如以下

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #include<cmath>
  5. #include<fstream>
  6. using namespace std;
  7. #define maxn 100
  8. ///最小堆调整函数
  9. void MinHeadfly(int *a,int i,int HeadSize)
  10. {
  11. int l=i*2,r=2*i+1;
  12. int largest;
  13. if(a[i]>a[l]&&l<=HeadSize)
  14. largest=l;
  15. else
  16. largest=i;
  17. if(a[largest]>a[r]&&r<=HeadSize)
  18. largest=r;
  19. if(largest!=i)
  20. {
  21. swap(a[i],a[largest]);
  22. MinHeadfly(a,largest,HeadSize);
  23. }
  24. }
  25. ///最小堆建立函数
  26. void MinHeadBuild(int *a,int n)
  27. {
  28. for(int i=n/2;i>=1;i--)
  29. MinHeadfly(a,i,n);
  30. }
  31. ///最小堆排序函数,从大到小排序
  32. void MinHeadSort(int *a,int HeadSize)
  33. {
  34. int k=HeadSize;
  35. for(int i=HeadSize;i>=2;i--)
  36. {
  37. swap(a[i],a[1]);
  38. k--;
  39. MinHeadfly(a,1,k);
  40. }
  41. }
  42. ///求b数组的前k大元素
  43. void prek(int *a,int *b,int n,int k)
  44. {
  45. MinHeadBuild(a,k);
  46. for(int i=k+1;i<=n;i++)
  47. if(b[i]>a[1])
  48. {
  49. a[1]=b[i];
  50. MinHeadfly(a,1,k);
  51. }
  52. MinHeadSort(a,k);
  53. cout<<"前"<<k<<"大元素为:"<<endl;
  54. for(int i=1;i<=k;i++)
  55. cout<<a[i]<<" ";
  56. cout<<endl;
  57. }
  58. int a[maxn],b[maxn];
  59. int main()
  60. {
  61. ifstream fin("lkl.txt");
  62. int n,k;
  63. //cout<<"请输入n,k: ";
  64. fin>>n>>k;
  65. //cout<<"请输入"<<n<<"个元素: "<<endl;
  66. for(int i=1;i<=n;i++)
  67. {
  68. fin>>b[i];
  69. if(i<=k)
  70. a[i]=b[i];
  71. }
  72. prek(a,b,n,k);
  73. return 0;
  74. }

算法导论学习之线性时间求第k小元素+堆思想求前k大元素的更多相关文章

  1. 算法导论 第八章 线性时间排序(python)

    比较排序:各元素的次序依赖于它们之间的比较{插入排序O(n**2) 归并排序O(nlgn) 堆排序O(nlgn)快速排序O(n**2)平均O(nlgn)} 本章主要介绍几个线性时间排序:(运算排序非比 ...

  2. 算法导论学习---红黑树具体解释之插入(C语言实现)

    前面我们学习二叉搜索树的时候发如今一些情况下其高度不是非常均匀,甚至有时候会退化成一条长链,所以我们引用一些"平衡"的二叉搜索树.红黑树就是一种"平衡"的二叉搜 ...

  3. 算法导论学习-Dynamic Programming

    转载自:http://blog.csdn.net/speedme/article/details/24231197 1. 什么是动态规划 ------------------------------- ...

  4. 算法导论学习-binary search tree

    1. 概念: Binary-search tree(BST)是一颗二叉树,每个树上的节点都有<=1个父亲节点,ROOT节点没有父亲节点.同时每个树上的节点都有[0,2]个孩子节点(left ch ...

  5. 算法---数组总结篇2——找丢失的数,找最大最小,前k大,第k小的数

    一.如何找出数组中丢失的数 题目描述:给定一个由n-1个整数组成的未排序的数组序列,其原始都是1到n中的不同的整数,请写出一个寻找数组序列中缺失整数的线性时间算法 方法1:累加求和 时间复杂度是O(N ...

  6. "《算法导论》之‘线性表’":基于静态分配的数组的顺序表

    首先,我们来搞明白几个概念吧(参考自网站数据结构及百度百科). 线性表 线性表是最基本.最简单.也是最常用的一种数据结构.线性表中数据元素之间的关系是一对一的关系,即除了第一个和最后一个数据元素之外, ...

  7. 算法导论学习-prim算法

    一. 关于最小生成树 对于无向连通图G=(V,E),其中V表示图的顶点,E表示图的边,对于每条边都有一个权值,可以理解为边a->b的权值C为从a走到b要走的路程为C.现在我们希望找到一个无回路的 ...

  8. 算法导论学习-RED-BLACK TREE

    1. 红黑树(RED-BLACK TREE)引言: ------------------------------------- 红黑树(RBT)可以说是binary-search tree的非严格的平 ...

  9. 算法导论学习-heapsort

    heap的定义:如果数组a[1,....n]满足:a[i]>a[2*i] && a[i]>a[2*i+1],1<=i<=n/2,那么就是一个heap,而且是ma ...

随机推荐

  1. IOI2008 Island 岛屿

    题目描述: bz luogu 题解: 裸的基环树直径. 代码: #include<queue> #include<cstdio> #include<cstring> ...

  2. 【实验吧】Just Click

    拿到答案需要正确地点击按钮 格式:simCTF{ } 解题链接: http://ctf5.shiyanbar.com/re/rev4.exe 由于最近在学数据库是c#编程,发现是c#,于是用.net ...

  3. C#sql语句如何使用占位符

    背景:在程序中,写sql语句时,可能要根据变量的值不同,SQL语句产生相应的变化.比如说存在变量StuName,根据变量值的不同,检索不同姓名的学生记录,这时需用到占位符的知识. 1,{0}占位符,代 ...

  4. tcpcopy简单用法

    这篇文章介绍下网易开源的流量重放(replay)工具TCPCopy,说是简单介绍,绝对不是谦虚,因为自己了解的确实也不多.为什么不甚了解呢,大家可以到TCPCopy的官方仓库看看,https://gi ...

  5. LA 4064 (计数 极角排序) Magnetic Train Tracks

    这个题和UVa11529很相似. 枚举一个中心点,然后按极角排序,统计以这个点为钝角的三角形的个数,然后用C(n, 3)减去就是答案. 另外遇到直角三角形的情况很是蛋疼,可以用一个eps,不嫌麻烦的话 ...

  6. kafka 理论学习

    http://blog.csdn.net/paul342/article/details/50479491 kafka基本知识.

  7. Leetcode 241.为运算表达式设计优先级

    为运算表达式设计优先级 给定一个含有数字和运算符的字符串,为表达式添加括号,改变其运算优先级以求出不同的结果.你需要给出所有可能的组合的结果.有效的运算符号包含 +, - 以及 * . 示例 1: 输 ...

  8. 2016上海站EC-final总结!

    2016上海站EC-final总结 本想在知乎上发起一个话题:没有半点准备实力菜得抠脚的选手突然有机会参加final是什么体验.不过感觉这样太高调了,于是..... 以上说的就是事实,毫无准备.毫无状 ...

  9. 运动员最佳匹配问题(km算法)

    洛谷传送门 带权二分图最大权完美匹配. 裸的km算法. 注意开long long. #include <cstdio> #include <cstring> #include ...

  10. 使用HttpWebRequest post数据时要注意UrlEncode

    今天在用HttpWebResponse类向一个远程页面post数据时,遇到了一个怪问题:通过对比自己post的参数和服务器接收到的值,发现参数中的一个+号被替换成了空格. 造成这个错误的原因在于+号在 ...