JS常见排序算法

JavaScript015

JS常见排序算法,第1张

排序算法说明:

(1)对于评述算法优劣术语的说明

稳定 :如果a原本在b前面,而a=b,排序之后a仍然在b的前面;

不稳定 :如果a原本在b的前面,而a=b,排序之后a可能会出现在b的后面;

内排序 :所有排序操作都在内存中完成;

外排序 :由于数据太大,因此把数据放在磁盘中,而排序通过磁盘和内存的数据传输才能进行;

时间复杂度 : 一个算法执行所耗费的时间。

空间复杂度 : 运行完一个程序所需内存的大小。

(2)排序算法图片总结:

1.冒泡排序:

解析:1.比较相邻的两个元素,如果前一个比后一个大,则交换位置。

 2.第一轮的时候最后一个元素应该是最大的一个。

 3.按照步骤一的方法进行相邻两个元素的比较,这个时候由于最后一个元素已经是最大的了,所以最后一个元素不用比较。

2.快速排序:

解析:快速排序是对冒泡排序的一种改进,第一趟排序时将数据分成两部分,一部分比另一部分的所有数据都要小。然后递归调用,在两边都实行快速排序。

3.插入排序:

解析:

 (1) 从第一个元素开始,该元素可以认为已经被排序

 (2) 取出下一个元素,在已经排序的元素序列中从后向前扫描

 (3) 如果该元素(已排序)大于新元素,将该元素移到下一位置

 (4) 重复步骤3,直到找到已排序的元素小于或者等于新元素的位置

 (5)将新元素插入到下一位置中

 (6) 重复步骤2

2.二分查找:

解析:二分查找,也为折半查找。首先要找到一个中间值,通过与中间值比较,大的放又,小的放在左边。再在两边中寻找中间值,持续以上操作,直到找到所在位置为止。

(1)递归方法

(2)非递归方法

4.选择排序:

解析:首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。

以此类推,直到所有元素均排序完毕。

5.希尔排序:

解析:先将整个待排序的记录序列分割成为若干子序列分别进行直接插入排序

6.归并排序:

解析:归并排序是一种稳定的排序方法。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。

7.堆排序:

解析:堆排序(Heapsort)是指利用堆这种数据结构所设计的一种排序算法。堆积是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是

小于(或者大于)它的父节点。

8.计数排序:

 解析:计数排序使用一个额外的数组C,其中第i个元素是待排序数组A中值等于i的元素的个数。然后根据数组C来将A中的元素排到正确的位置。它只能对整数进行排序。

9.桶排序:

解析:假设输入数据服从均匀分布,将数据分到有限数量的桶里,每个桶再分别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排

10.基数排序:

解析:基数排序是按照低位先排序,然后收集;再按照高位排序,然后再收集;依次类推,直到最高位。有时候有些属性是有优先级顺序的,先按低优先级排序,再按高优

先级排序。最后的次序就是高优先级高的在前,高优先级相同的低优先级高的在前。基数排序基于分别排序,分别收集,所以是稳定的。

基数排序 vs 计数排序 vs 桶排序

这三种排序算法都利用了桶的概念,但对桶的使用方法上有明显差异:

基数排序:根据键值的每位数字来分配桶 计数排序:每个桶只存储单一键值 桶排序:每个桶存储一定范围的数值

两者的功能没有什么区别,差别只是在于实现机制:

递归算法是通过递归来利用系统栈保存每次分割后左右序列的起始和终止位置

非递归算法则是用户自行编程用栈或者队列来保存每次分割后左右序列的起始和终止位置

如果是用栈,则非递归算法的严格执行过程也基本与递归算法一致

终于编写出来了,我写了两种,你看看:下面是代码:

/*非递归算法1

递归算法的开销很大,所以在下编了一个非递归的,算法描述如下:

A non-recursive version of quick sort using stack:

There are 2 stacks, namely one which stores the start of

a subarray and the other which stores the end of the

subarray.

STEP 1: while the subarray contains more than one element

,i.e. from Do {

SUBSTEP 1. pivot=Partion(subarray)

SUBSTEP 2. keep track of the right half of the current

subarray i.e. push (pivot+1) into stackFrom, push (to) into stackTo

SUBSTEP 3. go on to deal with the left half of

the current subarray i.e. to=pivot-1

}

STEP 2: if(neither of the stacks is empty)

Get a new subarray to deal with from the stacks.

i.e. start=pop(stackFrom)to=pop(stackTo)

STEP 3: both stacks are empty, and array has

been sorted. The program ends.

*/*/

void UnrecQuicksort(int q[],int low,int high)

{stack s1<br/>stacks2<br/>s1.push(low)<br/>s2.push(high)<br/>int tl,th,p<br/>while(!s1.empty() &&!s2.empty())<br/>{tl=s1.top()th=s2.top()<br/>s1.pop()s2.pop()<br/>if(tl>=th) continue<br/>p=partition(q,tl,th)<br/>s1.push(tl)s1.push(p+1)<br/>s2.push(p-1)s2.push(th)<br/>}

}

/*非递归算法2

要把递归算法改写成非递归算法,可引进一个栈,这个栈的大小取决于递归调用的深度,最

多不会超过n,如果每次都选较大的部分进栈,处理较短的部分,递归深度最多不超过log2n

,也就是说快速排序需要的附加存储开销为O(log2n)。

*/

void UnrecQuicksort2(int q[],int low,int high)

{int *a<br/>int top=0,i,j,p<br/>a=new int[high-low+1]<br/>if(a==NULL) return<br/>a[top++]=low<br/>a[top++]=high<br/>while(top>0)<br/>{i=a[--top]<br/>j=a[--top]<br/>while(j {p=partition(q,j,i)<br/>if(p-j {//先分割前部,后部进栈<br/>a[top++]=p+1<br/>a[top++]=i<br/>i=p-1<br/>}

else

{//先分割后部,前部进栈

a[top++]=j

a[top++]=p-1

j=p+1

}

}

}

}

/*打印输出*/

void display(int p[],int len)

{for(int i=0i cout<}

/*测试*/

int _tmain(int argc, _TCHAR* argv[])

{int a[]={49,65,97,12,23,41,56,14}

quicksort(a,0,7)

//UnrecQuicksort(a,0,7)

//UnrecQuicksort2(a,0,7)

display(a,8)

return 0

}