找到第k个最小元----快速选择
此算法借用快速排序算法。
这个快速选择算法主要利用递归调用,数组存储方式。包含3个文件,头文件QuickSelect.h,库函数QuickSelect.c,测试文件TestQuickSelect。
其中Cutoff可以自己给定,这个当开始给定的数组(或者递归调用产生的子数组)的元素个数<=20个时,采用插入排序。一般认为当元素个数<=20时,插入排序更快。这个20不是固定的,在这附近浮动都可以的。
头文件QuickSelect.h
#ifndef QuickSelect_H
#define QuickSelect_H
#define Cutoff 20
typedef int ElementType;
ElementType Median3(ElementType A[], int left, int right);
void Swap(ElementType *p, ElementType*q);
void InsertionSort(ElementType A[], int N);
ElementType QuickSelect(ElementType A[],int k, int N);
ElementType Qselect(ElementType A[],int k, int Left, int Right);
void PrintMatrix(ElementType A[], int N);
#endif // !QuickSelect_H
库函数QuickSelect.c
#include "QuickSelect.h"
#include<stdio.h>
//通过三数中值分割法获得数组的枢纽元
ElementType Median3(ElementType A[], int Left, int Right)
{
int Center = (Left + Right) / ;
if (A[Left] > A[Right])
Swap(&A[Left], &A[Right]);
if (A[Left] > A[Center])
Swap(&A[Left], &A[Center]);
if (A[Center] > A[Right])
Swap(&A[Center], &A[Right]);
//上述三次交换使得:A[Left]<A[Center]<A[Right]
Swap(&A[Center], &A[Right - ]);
return A[Right - ];//返回枢纽元;
} //交换指针p和q各自指向的值;
void Swap(ElementType * p, ElementType * q)
{
ElementType Tmp;
Tmp = *p;
*p = *q;
*q = Tmp;
} //当开始给定的数组(或者递归调用产生的子数组)的元素个数<=20个时,采用插入排序。
void InsertionSort(ElementType A[], int N)
{
int j, P;
ElementType Tmp;
for (P = ; P < N; P++)
{
Tmp = A[P];
for (j = P; j > && A[j - ] > Tmp; j--)
A[j] = A[j - ];
A[j] = Tmp;
}
} //快速选择算法驱动程序
ElementType QuickSelect(ElementType A[],int k, int N)
{
return Qselect(A, k-, , N - );//该程序即为,输入,k,返回A[k]。因为A从0开始存储,如果想找第5个最小值,其实找的是A[4],所以当我们找第k个值时,应输入k-1。
} //快速选择核心算法
ElementType Qselect(ElementType A[], int k, int Left, int Right)
{
int i, j;
ElementType Pivot;
if (Left + Cutoff - <= Right)//此处Cutoff-1是为了和N-1=Right对应上
{
Pivot = Median3(A, Left, Right);//调用枢纽元函数选取枢纽元
i = Left; j = Right - ;
for (; ;)
{
while (A[++i] < Pivot);
while (A[--j] > Pivot);
if (i < j) Swap(&A[i], &A[j]);
else break;
}
Swap(&A[i], &A[Right - ]);
if (k < i)
Qselect(A, k, Left, i - );//对剩下小于枢纽元的元素递归排序
else if (k>i)
Qselect(A, k, i + , Right);//对剩下大于枢纽元的元素递归排序
//当k=i时,即查询的值==枢纽元,则找到了,返回A[k]
}
else//当最后子数组只有Cutoff个,则采用插入排序。
InsertionSort(A + Left, Right - Left + );
return A[k];
} //打印数组
void PrintMatrix(ElementType A[], int N)
{
for (int i = ; i < N; i++)
printf("%d ", A[i]);
}
测试文件TestQuickSelect
#include "QuickSelect.h"
#include <stdio.h>
#include<time.h>
#include<stdlib.h>
#define N 100000
int main()
{
ElementType A[N] ;
int ID;
srand((unsigned)time(NULL));
for (int i = ; i < N; i++)
{
//A[i] = rand() % N;
A[i] =N-i;//100000个数,倒序排
} printf("\n");
//PrintMatrix(A, N);
printf("请输入要查询的ID:");
scanf_s("%d", &ID);
printf("Result:%d",QuickSelect(A, ID, N));
printf("\n");
}

找到第k个最小元----快速选择的更多相关文章
- 在未排序的数组中找到第 k 个最大的元素
在未排序的数组中找到第 k 个最大的元素.请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素. 示例 1: 输入: [3,2,1,5,6,4] 和 k = 2 输出: 5 ...
- 【一起刷LeetCode】在未排序的数组中找到第 k 个最大的元素
题目描述 在未排序的数组中找到第 k 个最大的元素.请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素. 示例 1: 输入: [3,2,1,5,6,4] 和 k = 2 ...
- 选取第K大数的快速选择算法和注意事项
快速选择算法,是一种能在大致O(N)的时间内选取数组中第k大或者k小的算法.其基本思路与快速排序算法类似,也是分治的思想. 其实这个算法是个基础算法,但是不常用,所以今天编的时候错了POJ2388,才 ...
- [程序员代码面试指南]第9章-在两个长度相等的排序数组中找到第k小的数(二分)
题目 给定两个有序数组arr1和arr2,再给定一个整数k,返回所有的数中第k小的数. 题解 利用题目"在两个长度相等的排序数组中找到第上中位数"的函数 分类讨论 k < 1 ...
- 比列的数目更多,以便找到第一k小值
问题叙述性说明:现有n作为一个有序序列(2,3,9),(3,5,11,23),(1,4,7,9,15,17,20),(8,15,35,9),(20,30,40),第k小值. 问题分析:可用多路归并排序 ...
- 215. Kth Largest Element in an Array【Medium】【找到第 k 大的元素】
Find the kth largest element in an unsorted array. Note that it is the kth largest element in the so ...
- 算法总结之 在两个排序数组中找到第K小的数
给定两个有序数组arr1 和 arr2 ,再给定一个int K,返回所有的数中第K小的数 要求长度如果分别为 N M,时间复杂度O(log(min{M,N}),额外空间复杂度O(1) 解决此题的方法跟 ...
- 【Leetcode 堆、快速选择、Top-K问题 BFPRT】有序矩阵中第K小的元素(378)
题目 给定一个 n x n 矩阵,其中每行和每列元素均按升序排序,找到矩阵中第k小的元素. 请注意,它是排序后的第k小元素,而不是第k个元素. 示例: matrix = [ [ 1, 5, 9], [ ...
- 【Leetcode 堆、快速选择、Top-K问题 BFPRT】数组中的第K个最大元素(215)
这道题很强大,引出了很多知识点 题目 在未排序的数组中找到第 k 个最大的元素.请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素. 示例 1: 输入: [3,2,1,5 ...
随机推荐
- [原] KVM 虚拟化原理探究(1)— overview
KVM 虚拟化原理探究- overview 标签(空格分隔): KVM 写在前面的话 本文不介绍kvm和qemu的基本安装操作,希望读者具有一定的KVM实践经验.同时希望借此系列博客,能够对KVM底层 ...
- 重撸js_2_基础dom操作
1.node 方法 返回 含义 nodeName String 获取节点名称 nodeType Number 获取节点类型 nodeValue String 节点的值(注意:文本也是节点) 2.inn ...
- MySQL碎碎念
1. 如何修改Mysql的用户密码 mysql> update mysql.user set password=password('hello') where user='root'; mysq ...
- DBA成长路线
从开发转为数据库管理,即人们称为DBA的已经有好几年,有了与当初不一样的体会.数据是企业的血液,数据是石油,数据是一切大数据.云计算的基础.作为DBA是数据的保卫者.管理者,是企业非常重要的角色.对于 ...
- [C#] C# 知识回顾 - 表达式树 Expression Trees
C# 知识回顾 - 表达式树 Expression Trees 目录 简介 Lambda 表达式创建表达式树 API 创建表达式树 解析表达式树 表达式树的永久性 编译表达式树 执行表达式树 修改表达 ...
- JavaScript基础知识总结(二)
JavaScript语法 二.数据类型 程序把这些量.值分为几大类,每一类分别叫什么名称,有什么特点,就叫数据类型. 1.字符串(string) 字符串由零个或多个字符构成,字符包括字母,数字,标点符 ...
- ASP.NET MVC——模型绑定
这篇文章我们来讲讲模型绑定(Model Binding),其实在初步了解ASP.NET MVC之后,大家可能都会产生一个疑问,为什么URL片段最后会转换为例如int型或者其他类型的参数呢?这里就不得不 ...
- JAVA构造时成员初始化的陷阱
让我们先来看两个类:Base和Derived类.注意其中的whenAmISet成员变量,和方法preProcess(). 情景1:(子类无构造方法) class Base { Base() { pre ...
- 多线程同步工具——Lock
本文原创,转载请注明出处. 参考文章: <"JUC锁"03之 公平锁(一)> <"JUC锁"03之 公平锁(二)> 锁分独占锁与共享锁, ...
- Android中的沉浸式状态栏效果
无意间了解到沉浸式状态栏,感觉贼拉的高大上,于是就是试着去了解一下,就有了这篇文章.下面就来了解一下啥叫沉浸式状态栏.传统的手机状态栏是呈现出黑色条状的,有的和手机主界面有很明显的区别.这一样就在一定 ...