第K小的数 快速排序 选择前K大O(n) 乱序数组中位数 median

2024-03-17 15:32

本文主要是介绍第K小的数 快速排序 选择前K大O(n) 乱序数组中位数 median,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

利用快速排序partition:

 

#include <iostream>
#include <map>
#include <algorithm>
#include <limits.h>
#include <assert.h>
using namespace std;
int selectK(int num[], int k, int left, int right) {assert(k <= (right - left + 1) && k >= 1);int mid = (left + right) / 2, i = left, j = right, pivot = num[mid];while (i <= j) {while (num[i] < pivot) {++i;}while (num[j] > pivot) {--j;}if (i <= j) {swap(num[i], num[j]);++i, --j;}}if (k == i - left && i - 1 == j + 1) {return pivot;}else if (k <= i - left) {return selectK(num, k, left, i-1);}else if (k > i - left) {return selectK(num, k - (i - left), i, right);}
}
int main() {int num[] = { 3, 2, 1, 4, 5, 6 };int res1 = selectK(num, 1, 0, 5);int res2 = selectK(num, 2, 0, 5);int res3 = selectK(num, 3, 0, 5);int res4 = selectK(num, 4, 0, 5);int res5 = selectK(num, 5, 0, 5);int res6 = selectK(num, 6, 0, 5);//int res6 = selectK(num, 6, 0, 4);return 0;
}

[l,i-1] <= pivot <= [j+1,r]

 

 

 

Python version:

from itertools import permutations#1. must confirm num[left, i-1] <= pivot <= num[j+1, right], that's why <= couldn't be replaced by < since it's impossible.
def selectKth(num, k, left, right):assert (k <= right-left+1 and k >= 1)mid = (left+right)//2i,j,pivot=left,right,num[mid]while (i <= j):     #2. must be <= instead of < since the returned condition required, otherwise, the recursive depth will be exceededwhile (num[i] < pivot): #3. must be < instead of <= since num[left, i-1] <= pivot <= num[j+1, right]i = i + 1while (num[j] > pivot): #3. must be > instead of >= since num[left, i-1] <= pivot <= num[j+1, right]j = j - 1if (i <= j):    #2. must be <= instead of < since the returned condition required, otherwise, the recursive depth will be exceedednum[i],num[j]=num[j],num[i]i,j = i+1,j-1#4. For num[left, i-1] <= pivot <= num[j+1, right], there're 2 cases for breaking the loop as the figure illustratedif (k == i-left and i-1==j+1 and pivot == num[i-1]):return pivotelif (k <= i-left):return selectKth(num, k, left, i-1)elif (k > i-left):return selectKth(num, k-(i-left), i, right)if __name__ == '__main__':perms = permutations([1,2,3,4,5,6], 6)for i in perms:for k in range(1, 7):perm = list(i)beforeSelect = list(i)kthNum = selectKth(perm, k, 0, 5)print("perm={0} k={1} kth={2}".format(beforeSelect, k, kthNum))assert (k == kthNum)

BTW, there's a O(n) method for selecting the first k numbers. Think over it!!!

from itertools import permutations# must confirm num[left, i-1] <= pivot <= num[j+1, right], that's why <= couldn't be replaced by < since it's impossible.def selectTopK(num, k, left, right):assert (k <= right - left + 1 and k >= 1)mid = (left + right) // 2i, j, pivot = left, right, num[mid]# must i <= j,while (i <= j):while (num[i] < pivot):i = i + 1while (num[j] > pivot):j = j - 1if (i <= j):num[i], num[j] = num[j], num[i]i, j = i + 1, j - 1# num[left, i-1] <= pivot <= num[j+1, right]if (k == i - left and i - 1 == j + 1 and pivot == num[i - 1]):return i - 1elif (k <= i - left):return selectTopK(num, k, left, i - 1)elif (k > i - left):return selectTopK(num, k - (i - left), i, right)if __name__ == '__main__':raw = [1, 2, 3, 1, 2, 3]rawlen = len(raw)perms = permutations(raw, rawlen)for i in perms:for k in range(1, rawlen+1):beforeSelect = list(i)kthIndex = selectTopK(beforeSelect, k, 0, rawlen-1)selectedTopK = beforeSelect[:kthIndex + 1]selectedTopK.sort()beforeSelect2 = list(i)beforeSelect2.sort()topK = beforeSelect2[:kthIndex + 1]print("perm={0} selected={1} topK={2}".format(beforeSelect, selectedTopK, topK))assert (topK == selectedTopK)

 

这篇关于第K小的数 快速排序 选择前K大O(n) 乱序数组中位数 median的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/819361

相关文章

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

使用Python快速实现链接转word文档

《使用Python快速实现链接转word文档》这篇文章主要为大家详细介绍了如何使用Python快速实现链接转word文档功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 演示代码展示from newspaper import Articlefrom docx import

Spring排序机制之接口与注解的使用方法

《Spring排序机制之接口与注解的使用方法》本文介绍了Spring中多种排序机制,包括Ordered接口、PriorityOrdered接口、@Order注解和@Priority注解,提供了详细示例... 目录一、Spring 排序的需求场景二、Spring 中的排序机制1、Ordered 接口2、Pri

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

Python 中 requests 与 aiohttp 在实际项目中的选择策略详解

《Python中requests与aiohttp在实际项目中的选择策略详解》本文主要介绍了Python爬虫开发中常用的两个库requests和aiohttp的使用方法及其区别,通过实际项目案... 目录一、requests 库二、aiohttp 库三、requests 和 aiohttp 的比较四、requ

Python中lambda排序的六种方法

《Python中lambda排序的六种方法》本文主要介绍了Python中使用lambda函数进行排序的六种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们... 目录1.对单个变量进行排序2. 对多个变量进行排序3. 降序排列4. 单独降序1.对单个变量进行排序

shell脚本快速检查192.168.1网段ip是否在用的方法

《shell脚本快速检查192.168.1网段ip是否在用的方法》该Shell脚本通过并发ping命令检查192.168.1网段中哪些IP地址正在使用,脚本定义了网络段、超时时间和并行扫描数量,并使用... 目录脚本:检查 192.168.1 网段 IP 是否在用脚本说明使用方法示例输出优化建议总结检查 1

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2