常见的排序算法

本文是lhyt本人原创,希望用通俗易懂的方法来理解一些细节和难点。转载时请注明出处。文章最早出现于本人github

0.前言

相信大家面试的时候都要经历手写什么什么排序这种事情吧,要不就是大概说一下思路。不许用各种语言封装好的函数、api,仅仅用原生的方法把他写出来。虽然看起来没什么意思,但是这也是考察一个人的基础有没有扎实、编程思想好不好的一种方法。

重要的事情说三遍:

主要理解快速排序!!!

主要理解快速排序!!!

主要理解快速排序!!!

而且要滚瓜烂熟!

以下所有的排序都是升序

1.冒泡排序

先说最简单的吧,冒泡,就是指数值大的,就是一个大气泡,慢慢冒到后面(水面)去。对于要排序的数组,一个个去遍历,大的数往后移。往后移怎么做到呢,就是遍历的时候,两个数中(第j和j+1个)要是谁比较大,谁在后面(交换位置)。气泡都聚集在水面(大数都从后面聚集在一起,所以每次第二层遍历都会到len-1-j截止)

时间复杂度O(n^2),如果是已经排好的情况是O(n)

function bubble(arr){

for(var i = 0,len = arr.length;i

for (var j = 0; i < len-1-j; j++) {

if(arr[j]>arr[j+1]){

var temp = arr[j+1]

arr[j] = arr[j+1]

arr[j] = temp

}

}

}

return arr

}

2.快速排序(重点)

快排是面试问得最多的了,也是目前用得最多,效率最稳的方法。快速排序的最慢情况是O(n²),升序的时候。但它的数学期望是O(n log n) ,且O(n log n)记号中隐含的常数因子很小,比复杂度稳定等于O(n log n)的归并排序要小很多。对绝大多数顺序性较弱的随机数列而言,快速排序总是占优。

首先,取一个基准值(我们取第一个,也可以取中间、取最后都行),接着让小于基准值的在左边,大于的在右边,分成左右两堆。然后分别对左边和右边那堆采取同样的操作,取基准值,让基准值左边都是小于他的,右边都是大于他的。一直重复操作,直到全部升序。

2.1另外开辟两个空数组

这种方法容易理解,但是依赖另外开辟出来的数组。我们取第一个作为基准,从第二(特别注意,第二个开始!不然栈溢出)个开始遍历,小于基准的数放在左数组(left),大于基准的放在右数组,接着对左数组和右边数组重复操作,再对左边数组的左边和右边进行操作......直到数组长度为1

function quick(arr){

if(arr.length <= 1){

return arr

}

var pivot = arr[0]

var left = []

var right = []

var len = arr.length

for(var i = 1;i

if(arr[i]>pivot){

right.push(arr[i])

}else{

left.push(arr[i])

}

}

return quick(left).concat([pivot],quick(right))

}

当然网上也有人家从中间开始的,中间开始的话,就用splice去除这个值,再给pivot赋值

2.2直接在原数组操作

这种比较难理解,但是不会占用其他内存。基准值也是取第一个,quick传入3个参数:数组,子数组的起始位置、子数组的结束位置。第一次赋值肯定是quick(arr),left和right是undefined,left和right默认是第一个和最后一个数的索引。 partitionIndex是基准值索引。

核心部分是中间判断left

取第一个基准值,从第二个(index)开始遍历。index是目前最后一个小于基准值的索引(其实是留给基准值自己的一个空位)。如果遍历中第i个有小于基准值的,i与index互换,index再+1,这是什么效果呢,可以看一下:8,5,10,7,6的演示

基准值是8,从5开始遍历(var i = index; i <= right; i++)。

i=1发现5<8,arr【1】与arr【1】交换位置,index+1,此时数组不变8,5,10,7,6,但是index已经是2

i=2发现10>8,跳过,index=2

i=3发现7<8,arr【3】与arr【2】交换,8,5,7,10,6,index=3

i=4发现6<8,arr【4】与arr【3】交换,8,5,7,6,10,index=4

遍历已经完成,把基准值和最后一个小于他的arr【index-1】互换6,5,7,8,10,已经完成了基准值左边都是小于他的、右边都是大于他的目的。

一个模块的操作完成,基准值索引partitionIndex变成index-1

接下来就是对分出来的左数组和右数组重复的递归操作

function quick(arr, left, right) {

var len = arr.length,

partitionIndex,

left = typeof left == 'number' ? left : 0,

right = typeof right == 'number' ? right : len-1;

if (left < right) {//保证小于基准值的在左边,大的在右边

var pivot = left, //基准值是数组的第一个

index = pivot + 1; //从数组的第二个开始

for (var i = index; i <= right; i++) {

if (arr[i] < arr[pivot]) {

var temp = arr[i];

arr[i] = arr[index];

arr[index] = temp;

index++;

}

}

var temp = arr[pivot];

arr[pivot] = arr[index - 1];

arr[index - 1] = temp;

partitionIndex = index-1;

quick(arr, left, partitionIndex-1);

quick(arr, partitionIndex+1, right);

}

return arr;

}

3.选择排序

表现最稳定的排序之一,无论什么数据进去都是O(n²)复杂度,但是依赖额外内存保存最小值

数组长度为len的数组中进行选择排序时:

取后len个的最小值放数组第一个位置

取后len-1个的最小值放数组第二个位置

取后len-2个的最小值放数组第三个位置

.......

显然,最后一个就是最大的,前面的已经完成了排序

function select(arr){

var min //保存运算过程中最小值索引

var temp

for (var i = 0,len = arr.length; i

min = i //先默认最小是自己

for(var j = i+1;j

if(arr[min]>arr[j]){

min = j //记录更小的值的索引

}

}

temp = arr[i]

arr[i] = arr[min]

arr[min] = temp

}

return arr

}

4.插入排序

就像打牌一样,自己是怎么整理的呢。拿着一张牌,一个个往下去对比,发现下一个大于(或等于)自己,上一个小于(或等于)自己,那就把他放在中间。复杂度稳定O(n^2),最好的情况是已经排序完成时O(n),依赖额外内存,保存当前遍历的数

function insert(arr) {

var len = arr.length;

var preIndex, current;

for (var i = 1; i < len; i++) {

preIndex = i - 1;//小于自己的数的索引

current = arr[i];

while(preIndex >= 0 && arr[preIndex] > current) {

arr[preIndex+1] = arr[preIndex];//如果满足条件,把当前的数插入中间

preIndex--;

}

arr[preIndex+1] = current;

}

return arr;

}

5.计数排序

将数组区间取出来(【min,max】),然后对数组进行遍历,计算每一个数组元素在区间【min,max】中出现次数,最后从头到尾把数组写出来。复杂度是O(n),比较稳定,但是依赖额外内存空间

计数排序需要全部都是整数!

这里,我们取正整数来试验,所以只要取得最大值就行。arr的值表示bucket的位置,如果bucket的某个位置是undefined,说明arr里面没有这个位置的索引值

function count(arr) {

var max = Math.max(...arr)//需要知道最大值的,这里只能作弊了

var bucket = new Array(max+1),//一个被max+1个undefined填充的数组

sortedIndex = 0;//重写arr的索引

for (var i = 0; i < arr.length; i++) {

if (!bucket[arr[i]]) {//第一次遇到bucket的某个索引值为arr值,将undefined变成0

bucket[arr[i]] = 0;

}

bucket[arr[i]]++;//开始统计

}

for (var j = 0; j < max + 1; j++) {

while(bucket[j] > 0) {

arr[sortedIndex++] = j;//重写arr

bucket[j]--;//再一次进入同样的循环,直到没有重复值(0的时候)

}

}

return arr;

}

6.基数排序

先比较个位数的大小,按顺序分类,从头到尾重新取出数字。再进行第二次比较,比较十位数的大小,按顺序分类,再从头到尾取出........直到最高位。要求全是整数。

我们这里比较两位数以内的

var counter = [];

function radixSort(arr, maxDigit) {

var mod = 10;

var dev = 1;

for (var i = 0; i < maxDigit; i++, dev *= 10, mod *= 10) {

for(var j = 0; j < arr.length; j++) {

var bucket = parseInt((arr[j] % mod) / dev);

if(counter[bucket]==null) {

counter[bucket] = [];

}

counter[bucket].push(arr[j]);//和计数排序一样,这里只需要0-9的数组存放

}

var pos = 0;

for(var j = 0; j < counter.length; j++) {

var value = null;

if(counter[j]!=null) {

while ((value = counter[j].shift()) != null) {

arr[pos++] = value;

}

}

}

}

return arr;

}

7.归并排序

由名字可以知道,这其实就是一种递归。顺序是比较前面两个=>比较前面两个的后面两个=>比较前面四个=>比较前面四个的后面两个=>比较前面6个的后面两个=>比较前面8个........

始终都是O(nlogn)的复杂度,不受输入数据影响,但是依赖额外内存

每一次2个数的小组比较,两个小组的数量一致而且排序方式也是升序,对比起来就是一一对比。大的组比较也是一样。数量不同只有数组长度非2的n次方的尾部的比较。

function mergeSort(arr) { //采用自上而下的递归方法

var len = arr.length;

if(len < 2) {

return arr;//递归到子数组长度为1为止

}

var middle = Math.floor(len / 2),//数组被分成左右两个子数组

left = arr.slice(0, middle),

right = arr.slice(middle);

return merge(mergeSort(left), mergeSort(right));

}

function merge(left, right)

{

var result = [];

while (left.length && right.length) {

if (left[0] <= right[0]) {//两个子数组从头开始一一比较,归并为一个排序好的数组

result.push(left.shift());

} else {

result.push(right.shift());

}

}

while (left.length)

result.push(left.shift());//取出第一个取比较

while (right.length)

result.push(right.shift());

return result;

}


原文来源于:lhyt的github

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,194评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,058评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,780评论 0 346
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,388评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,430评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,764评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,907评论 3 406
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,679评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,122评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,459评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,605评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,270评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,867评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,734评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,961评论 1 265
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,297评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,472评论 2 348

推荐阅读更多精彩内容

  • 某次二面时,面试官问起Js排序问题,吾绞尽脑汁回答了几种,深感算法有很大的问题,所以总计一下! 排序算法说明 (1...
    流浪的先知阅读 1,189评论 0 4
  • 排序算法说明 (1)排序的定义:对一序列对象根据某个关键字进行排序; 输入:n个数:a1,a2,a3,…,an 输...
    code武阅读 653评论 0 0
  • 总结一下常见的排序算法。 排序分内排序和外排序。内排序:指在排序期间数据对象全部存放在内存的排序。外排序:指在排序...
    jiangliang阅读 1,334评论 0 1
  • 在很多时候,我们生活在这个世界都在不停的寻找意义,寻找真相,寻找一些深层次的东西,这些东西是不流于表面的,是意义深...
    眼睛睁_闫老湿阅读 195评论 12 3
  • 他老了。他的修鞋机也老了。 他安静地坐在农贸市场的角落。皮鞋、休闲鞋、凉鞋,破损的地方都在告诉他,这个世界的节奏在...
    中正恩泽阅读 233评论 0 4