内容:

1、原始问题     =》  O(N*logN)

2、BFPRT算法    =》 O(N)

1、原始问题

问题描述:给你一个整型数组,返回其中第K小的数

普通解法:

这道题可以利用荷兰国旗改进的 partition 和随机快排的思想:随机选出一个数,将数组以该数作比较划分为 <,=,> 三个部分,

则 = 部分的数是数组中第几小的数不难得知,接着对 < (如果第K小的数在 < 部分)或 > (如果第K小的数在 > 部分)部分的数

递归该过程,直到 = 部分的数正好是整个数组中第K小的数。这种做法不难求得时间复杂度的数学期望为 O(NlogN) (以2为底)。

但这毕竟是数学期望,在实际工程中的表现可能会有偏差(最坏情况下的时间复杂度会达到O(N^2))

BFPRT算法可以说是这种算法的一种优化吧,故在此就不写这种解法的代码了

另外一种普通解法:

用堆去做,时间复杂度是靠谱的O(N*logk)

代码如下:

     // 大根堆比较器
public static class MaxheapComparator implements Comparator<Integer> {
@Override
public int compare(Integer o1, Integer o2) {
return o2 - o1;
}
} // O(N*logk)的解法
public static PriorityQueue getMinKNumsByHeap(int[] arr, int k) {
if (k < 1 || k > arr.length) {
return null;
}
PriorityQueue<Integer> kHeap = new PriorityQueue<Integer>(k,
new MaxheapComparator());
for (int i = 0; i != k; i++) {
kHeap.add(arr[i]);
}
for (int i = k; i != arr.length; i++) {
if (arr[i] < kHeap.peek()) {
kHeap.poll();
kHeap.add(arr[i]);
}
}
return kHeap;
} public static void main(String[] args) {
int[] arr = { 1, 3, 2, 5, 9 };
// 测试普通方法
System.out.println(getMinKNumsByHeap(arr, 1).peek());
System.out.println(getMinKNumsByHeap(arr, 2).peek());
System.out.println(getMinKNumsByHeap(arr, 3).peek());
System.out.println(getMinKNumsByHeap(arr, 4).peek());
System.out.println(getMinKNumsByHeap(arr, 5).peek());
}

2、BFPRT算法

BFPRT算法能够做到时间复杂度就是 O(N)    BFPRT算法,接收一个数组和一个K值,返回数组中的一个数

1. 数组被划分为了 N/5 个小部分,每个部分的5个数排序需要 O(1) ,所有部分排完需要 O(N/5)=O(N)

2. 取出每个小部分的中位数,一共有 N/5 个,递归调用BFPRT算法得到这些数中第 (N/5)/2 小的数(即这些数 的中位数),记为 pivot

3. 以 pivot 作为比较,将整个数组划分为 <pivot , =pivot , >pivot 三个区域

4. 判断第K小的数在哪个区域,如果在 = 区域则直接返回 pivot ,如果在 < 或 > 区域,则将这个区域的数递 归调用BFPRT算法

5. base case :在某次递归调用BFPRT算法时发现这个区域只有一个数,那么这个数就是我们要找的数

     // O(N)的解法
public static int getMinKthNum(int[] arr, int k){
if(arr==null||k>arr.length){
return Integer.MIN_VALUE;
}
int[] copyArr = Arrays.copyOf(arr, arr.length);
return BFPRT(copyArr, 0, arr.length-1, k-1);
} private static int[] partition(int[] arr, int begin, int end, int pivot){
int L = begin-1;
int R = end + 1;
int cur = begin;
while(cur!=R){
if(arr[cur]>pivot){
swap(arr, cur, --R);
} else if(arr[cur]<pivot){
swap(arr, cur++, ++L);
} else{
cur++;
}
}
return new int[]{L+1, R-1};
} private static int BFPRT(int[] arr, int begin, int end, int i) {
if (begin == end) {
return arr[begin];
}
int pivot = medianOfMedians(arr, begin, end);
int[] pivotRange = partition(arr, begin, end, pivot);
if(i>=pivotRange[0]&&i<=pivotRange[1]){
return arr[i];
} else if(i<pivotRange[0]){
return BFPRT(arr, begin, pivotRange[0]-1, i);
} else{
return BFPRT(arr, pivotRange[1] + 1, end, i);
}
} private static int medianOfMedians(int[] arr, int begin, int end) {
int num = end - begin + 1;
int offset = num % 5 == 0 ? 0 : 1;
int[] medians = new int[num / 5 + offset];
for (int i = 0; i < medians.length; i++) {
int beginI = begin + i * 5;
int endI = beginI + 4;
medians[i] = getMedian(arr, beginI, Math.min(endI, end));
}
return BFPRT(medians, 0, medians.length - 1, medians.length / 2);
} private static int getMedian(int[] arr, int begin, int end){
insertionSort(arr, begin, end);
int sum = end + begin;
int mid = (sum/2) + (sum%2);
return arr[mid];
} private static void insertionSort(int[] arr, int begin, int end){
if(begin>=end){
return;
}
for(int i=begin+1;i<=end;i++){
for(int j=i;j>begin;j--){
if(arr[j]<arr[j-1]){
swap(arr, j, j-1);
} else{
break;
}
}
}
} private static void swap(int[]arr , int i, int j){
int tmp = arr[i];
arr[i] = arr[j];
arr[j] = tmp;
}

经典算法 BFPRT算法详解的更多相关文章

  1. JVM垃圾回收算法及回收器详解

    引言 本文主要讲述JVM中几种常见的垃圾回收算法和相关的垃圾回收器,以及常见的和GC相关的性能调优参数. GC Roots 我们先来了解一下在Java中是如何判断一个对象的生死的,有些语言比如Pyth ...

  2. 【机器学习】【条件随机场CRF-2】CRF的预测算法之维特比算法(viterbi alg) 详解 + 示例讲解 + Python实现

    1.CRF的预测算法条件随机场的预测算法是给定条件随机场P(Y|X)和输入序列(观测序列)x,求条件概率最大的输出序列(标记序列)y*,即对观测序列进行标注.条件随机场的预测算法是著名的维特比算法(V ...

  3. 最短路径Floyd算法【图文详解】

    Floyd算法 1.定义概览 Floyd-Warshall算法(Floyd-Warshall algorithm)是解决任意两点间的最短路径的一种算法,可以正确处理有向图或负权的最短路径问题,同时也被 ...

  4. openerp经典收藏 字段定义详解(转载)

    字段定义详解 原文地址:http://shine-it.net/index.php/topic,2159.0.htmlhttp://blog.sina.com.cn/s/blog_57ded94e01 ...

  5. openerp经典收藏 对象定义详解(转载)

    对象定义详解 原文地址:http://shine-it.net/index.php/topic,2159.0.htmlhttp://blog.sina.com.cn/s/blog_57ded94e01 ...

  6. 一个经典的 HTTP协议详解

    1引言 HTTP是一个属于应用层的面向对象的协议,由于其简捷.快速的方式,适用于分布式超媒体信息系统.它于1990年提出,经过几年的使用与发展,得到不断地完善和扩展.目前在WWW中使用的是HTTP/1 ...

  7. c++ LeetCode(初级数组篇)十一道算法例题代码详解(一)

    原文作者:aircraft 原文链接:https://www.cnblogs.com/DOMLX/p/10940636.html 唉!最近忙着面试找实习,然后都是面试的很多是leetcode的算法题, ...

  8. KMP算法 Next数组详解

    题面 题目描述 如题,给出两个字符串s1和s2,其中s2为s1的子串,求出s2在s1中所有出现的位置. 为了减少骗分的情况,接下来还要输出子串的前缀数组next.如果你不知道这是什么意思也不要问,去百 ...

  9. Dijkstra算法之 Java详解

    转载:http://www.cnblogs.com/skywang12345/ 迪杰斯特拉算法介绍 迪杰斯特拉(Dijkstra)算法是典型最短路径算法,用于计算一个节点到其他节点的最短路径. 它的主 ...

随机推荐

  1. Flume-NG源码阅读之SpoolDirectorySource(原创)

    org.apache.flume.source.SpoolDirectorySource是flume的一个常用的source,这个源支持从磁盘中某文件夹获取文件数据.不同于其他异步源,这个源能够避免重 ...

  2. Html页面Dom对象之Element

    HTML DOM Element 对象 HTML DOM 节点 在 HTML DOM (文档对象模型)中,每个部分都是节点: 文档本身是文档节点 所有 HTML 元素是元素节点 所有 HTML 属性是 ...

  3. Thrift 个人实战--Thrift 网络服务模型(转)

    前言: Thrift作为Facebook开源的RPC框架, 通过IDL中间语言, 并借助代码生成引擎生成各种主流语言的rpc框架服务端/客户端代码. 不过Thrift的实现, 简单使用离实际生产环境还 ...

  4. 转 Katana 项目入门

    Katana 项目入门 Howard Dierking 当 ASP.NET 首次在 2002 年发布时,时代有所不同. 那时,Internet 仍处于起步阶段,大约有 5.69 亿用户,每个用户平均每 ...

  5. java打jar包的几种方式详解

    经常会头疼于一个jar包是如何制作的,包括maven的打包方式,springboot的打jar包的原理,jar包稍稍有错误就会完全无法运行.在网上折腾了很久终于有些思路和步骤,在这里做个笔记 本文目录 ...

  6. Excel 从字符串中提取日期值

    因为工作需要,Excel 表中有一串字符,需要将字符里的日期提取出,并转成日期值. 需要转成如下格式: 可使用以下公式. =DATEVALUE(TEXT(MID(I2,1,4)+1&" ...

  7. storm之 Storm 工作原理

    Storm 工作原理 Storm简介 1.Storm是一套分布式的.可靠的,可容错的用于处理流式数据的系统. 2.Storm也是基于C/S架构来进行工作的,C负责将数据处理的方式的jar(Topolo ...

  8. Oracle数据泵的使用

    几乎所有DBA都熟悉oracle的导出和导入实用程序,它们将数据装载进或卸载出数据库,在oracle  database 10g和11g中,你必须使用更通用更强大的数据泵导出和导入(Data Pump ...

  9. velocity 知识点

    velocity 教程: http://www.51gjie.com/javaweb/126 velocity 语法 语法 说明 关键字以#开头 定义数组 ['aaa','bbb'] 变量以$开头 把 ...

  10. vue-cli 中的静态资源处理

    你会注意到在项目结构上我们有静态资源两个目录:src/assets 和 static/.它们之间有什么区别? 1. 通过webpack处理的资源 要回答这个问题,我们首先需要了解webpack如何处理 ...