1、选择排序
首先在未排序数列中找到最小元素
,然后将其与数列的首部元素
进行交换,然后,在剩余未排序元素中继续找出最小元素,将其与已排序数列的末尾位置元素交换。以此类推,直至所有元素均排序完毕.复杂度为n2,复杂度还是相当高的
选择排序即是采用交换次数最少的方针进行排序,选择排序的交换次数要远远少于冒泡排序
力扣2471逐层排序二叉树所需的最少操作数目用到了选择排序算法
选择排序算法是不稳定的,值相同的元素的相对位置可能会发生改变,这一点需要特别注意
2、快速排序
快速排序是不稳定的
快速排序的时间复杂度在最坏情况下是O(N2),平均的时间复杂度是O(N*lgN)。
代码及注解如下所示
public void sort(int arr[],int begin,int end){
if(begin>=end){
return;
}
//左边的i,左指针
int start_pos=begin;
//右边的j,右指针
int end_pos=end;
//基准值,默认为第一个元素
int standard=arr[start_pos];
while (start_pos<end_pos){
//右指针开始往左移,寻找比基准值小的元素
while (start_pos<end_pos&&standard<=arr[end_pos]){
end_pos--;
}
//如果确实找到该元素,那么就进行置换,置换过后左指针右移一位
if(arr[end_pos]<standard){
arr[start_pos]=arr[end_pos];
start_pos++;
}
//左指针开始往右移动,寻找比基准值大的元素
while (start_pos<end_pos&&standard>=arr[start_pos]){
start_pos++;
}
//如果确实找到该元素,那么就进行置换,置换过后右指针左移一位
if(arr[start_pos]>standard){
arr[end_pos]=arr[start_pos];
end_pos--;
}
}
//左右指针重叠,将基准值赋予左右指针重叠处
arr[start_pos]=standard;
//分治递归
sort(arr,begin,start_pos-1);
sort(arr,start_pos+1,end);
}
3、堆排序
什么是堆?
堆的实现方式:
数组按顺序存储层序遍历的二叉树,按顺序全部放进来,因为是完全二叉树,数组中间不可能存在空缺
用数组存储这样的好处便是,已知一个节点在数组中的索引为k时,它的父节点位置为k/2,它的两个子结点的位置分别为2k和2k+1.如此这般,在不使用指针的情况下,也可以通过计算数组的索引在树中上下移动,从a[k]向上一层,就令k=k/2,向下一层就令k等于2k或2k+1
堆的特性:
对于每个节点的值都大于等于子树中每个节点值的堆,我们叫做“大顶堆”。对于每个节点的值都小于等于子树中每个节点值的堆,我们叫做“小顶堆”。
堆的插入:
插入新节点s,插入节点后,堆不一定还是标准的堆,需要将插入的元素不断向父节点的值去做比较,如果不符合条件,则交换两者的值,直到父节点的值与子节点的关系符合条件为止,如图所示:
堆的删除:
以删除根节点为例,将堆中最后一个元素与根节点元素互换,如图所示
将置换后的根节点删除,如图所示,
此时堆并不是标准的堆,要通过不断调整使得堆变成标准堆(下沉操作),通过逐次下沉操作使得堆变为标准堆
堆基本操作的代码如下所示,包含堆的插入节点、删除节点、上浮、下沉等算法
public class Heap<T extends Comparable<T>>{
//存储堆中的元素
private T[] items;
//记录堆中元素的个数
private int N;
//初始化,数组中的0索引处已经废弃掉了
public Heap(int capacity){
this.items=(T[]) new Comparable[capacity+1];
this.N=0;
}
//判断堆中索引i处的元素是否小于索引j处的元素
private boolean less(int i,int j){
return items[i].compareTo(items[j])<0;
}
//交换堆中i索引和j索引处的值
private void exch(int i,int j){
T temp=items[i];
items[i]=items[j];
items[j]=temp;
}
//往堆中插入一个元素
public void insert(T t){
items[++N]=t;
//使得插入的元素处于正确的位置
swim(N);
}
//使用上浮算法,使得索引k处的元素能在堆中处于一个正确的位置
public void swim(int k){
//通过循环,不断比较当前节点的值和父节点的值,如果发现父节点的值比当前节点小,则交换位置
while (k>1){
//比较当前节点和其父节点,如果小,则交换两者的位置
if(less(k/2,k)){
exch(k/2,k);
}
k=k/2;
}
}
//删除堆中最大的元素,并返回这个最大元素
public T delMax(){
T max=items[1];
//交换索引1处的元素和最大索引处的元素,让完全二叉树的最右侧元素变成临时根节点
exch(1,N);
//删除最大索引处的节点
items[N]=null;
//元素个数减一
N--;
//通过下沉调整堆,让堆重新有序
sink(1);
return max;
}
//使用下沉算法
public void sink(int k){
//通过循环不断对比当前k节点及其左子节点2k、右子节点2k+1,如果当前节点小,则需要交换位置
//存在删除后不存在右子节点的情况
//-----------------这里的范围非常关键
while (2*k<=N){
//-------------------
//获取当前节点的较大子节点值
int max;//记录较大节点所在的索引
if(2*k+1<=N){
if(less(2*k,2*k+1)){
max=2*k+1;
}
else {
max=2*k;
}
}
else {
max=2*k;
}
//如果已经符合条件
if(!less(k,max)){
break;
}
//如果不符合条件,则交换k和max索引处的值
exch(k,max);
//k往下走
k=max;
}
}
}
堆排序:
1、根据原数组构造堆
即扫描前一半的数组元素,进行下沉操作,即可构造出标准的堆
2、采用类似堆删除操作,实现堆内的排序
代码实现如下
public class HeapSort{
//判断heap堆中索引i处的元素是否小于索引j处的值
private boolean less(Comparable[]heap,int i,int j){
return heap[i].compareTo(heap[j])<0;
}
//交换heap堆中i索引和j索引处的值
private void exch(Comparable[]heap,int i,int j){
Comparable tmp=heap[i];
heap[i]=heap[j];
heap[j]=tmp;
}
//根据原数组source,构造出堆heap
private void createHeap(Comparable[] source,Comparable[]heap){
//把source中的元素拷贝到heap中,heap中的元素就形成一个无序的堆
System.arraycopy(source,0,heap,1,source.length);
//对堆中的元素做下沉调整(从长度的一半开始,往1处扫描)
for(int i=(heap.length)/2;i>0;i--){
sink(heap,i,heap.length-1);
}
}
//对source数组中的数据从小到大排序
private void sort(Comparable[] source){
//不断地将根节点与最大索引处的值进行交换,交换过后通过下沉算法重新使得堆变得有序!
//构建堆
Comparable[] heap = new Comparable[source.length + 1];
createHeap(source,heap);
//定义一个变量,记录未排序的元素中最大的索引
int N=heap.length-1;
//通过循环,交换索引1处的元素和排序的元素中最大的索引处的元素
while (N!=1){
exch(heap,1,N);
//排序交换后最大元素所在的索引,让它不要参与堆的下沉调整
N--;
//需要对索引1处的元素进行下沉调整
sink(heap,1,N);
}
//把heap中的数据复制到原数组source中
System.arraycopy(heap,1,source,0,source.length);
}
//在heap中,对target处的元素做下沉,范围是0-range
private void sink(Comparable[]heap,int target,int range){
while ((2*target<=range)){
int max;
//1.找出当前节点的较大子节点
if(2*target+1<=range){
if(less(heap,2*target,2*target+1)){
max=2*target+1;
}
else {
max=2*target;
}
}else {
max=2*target;
}
//2、比较当前节点的值和较大子节点的值
if(!less(heap,target,max)){
break;
}
exch(heap,target,max);
target=max;
}
}
}
4、归并排序:
思路如下图所示,将数组不断拆分,直至只剩两个元素时进行大小的比较,从而实现两个有序数组的合并,合并成一个有序的数组;采用了分治的思想;
归并排序是稳定的排序,时间复杂度为O(nlogn)空间复杂度为O(n)
public static void merge(int arr[],int start,int mid,int end){
int temp[]=new int[end-start+1];
int left=start;
int right=mid+1;
int index=0;
while (left<=mid&&right<=end){
if(arr[left]<=arr[right]){
temp[index]=arr[left];
left++;
}
else {
temp[index]=arr[right];
right++;
}
index++;
}
while (left<=mid){
temp[index]=arr[left];
index++;
left++;
}
while (right<=end){
temp[index]=arr[right];
index++;
right++;
}
index=0;
for(int i=start;i<=end;i++){
arr[i]=temp[index];
index++;
}
}
public static void mergesort(int arr[],int start,int end){
if(start>=end){
return;
}
int mid=(start+end)/2;
mergesort(arr,start,mid);
mergesort(arr,mid+1,end);
merge(arr,start,mid,end);
}
4、冒泡排序:依次比对相邻元素,如果相邻元素不符合升序/降序排列,则进行交换
代码:
public static void bubblesort(int arr[]){
for(int i=arr.length-1;i>=1;i--){
for(int j=0;j<i;j++){
if(arr[j]>arr[j+1]){
int temp=arr[j];
arr[j]=arr[j+1];
arr[j+1]=temp;
}
}
}
}
5、选择排序:不断的去寻找子数组最小值,找到后和头元素进行互换