Java经典面试题——手写快速排序和归并排序

发布时间:2023年12月23日

题目链接:https://www.luogu.com.cn/problem/P1177
在这里插入图片描述

输入模板:

5
4 2 4 5 1

快速排序

技巧:交换数组中的两个位置

a[l] = a[l] + a[r] - (a[r] = a[l]);

稳定不稳定?:不稳定

注意找哨兵那里内循环的等于号不能漏,不然能出不来循环了,因为如果数值都一样,那么l和r一直保持不变了

import java.util.*;

public class Main {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int n = scanner.nextInt();
        int[] a = new int[1000005];
        for(int i = 1; i<= n;i++)
            a[i] = scanner.nextInt();
        scanner.close();
        quickSort(a,1,n);
        for(int i=1;i<=n;i++){
            if(i==1) 
                System.out.print(a[i]);
            else
                System.out.print(" "+a[i]);
        }
        return ;
    }
    //数组的传递时通过引用进行传递的
    public static void quickSort(int[] a, int left, int right){
        //设置一个哨兵,分治排序,使得小于哨兵的所有节点都小于他,大于哨兵的所有
        //节点都大于他
        if(left>=right)
            return ; 
        int solder = partSort(a, left, right);
        quickSort(a, left, solder-1);
        quickSort(a, solder+1, right);
        return ;
    }
    public static int partSort(int[] a,int left,int right){
        int pivot=a[left];
		 while(left<right) {
			 //起初,一定要从右边指针开始,因为arr[low]的值已经扔给了pivot,arr[low]
			 //想象成无数字的空位
			 while(left<right&&pivot<=a[right])
				 --right;
			 //把比pivot的小的数扔到左边指针
			 //把arr[high]扔到arr[low]这个空位上
			 //然后,high位置可以想象成无数字的空位
			 a[left]=a[right];
			 while(left<right&&a[left]<=pivot)
				 ++left;
			 //把比pivot大的数扔到右边
			//把arr[low]扔到arr[high]这个空位上
			//然后,low位置可以想象成是无数字的空位
			 a[right]=a[left];
		 }
		 //此时low==high,return high也一样
		 a[left]=pivot;
		return left;
    }   
}

归并排序

归并比快排好写

import java.util.*;

public class Main {
    public static void main(String[] args) {
        //手写一个快排,分治排序
        Scanner scanner = new Scanner(System.in);
        int n = scanner.nextInt();
        int[] array = new int[n];
        for(int i = 0;i<=n-1;i++)
            array[i] = scanner.nextInt();
        QuickSort(array,0,n-1);
        for(int x:array)
            System.out.print(x+" ");
    }
    public static void QuickSort(int[] array,int left, int right){
        //System.out.println(left+" "+right);
        if(left>=right)
            return ;
        else{
            int mid = (left+right)/2;
            QuickSort(array, left, mid);
            QuickSort(array, mid+1, right);
            MergeArray(array,left,right,mid+1);
        }
        return ;
    }
    public static void MergeArray(int[] array,int left,int right,int mid){
        int[] temp = new int[array.length];
        int p = left,q = mid;
        for(int i=left;i<=right;i++){
            if(q>right||p<mid&&array[p]<array[q]){
                temp[i]=array[p];
                p++;
            }else{
                temp[i]=array[q];
                q++;
            }
        }
        for(int i=left;i<=right;i++){
            array[i] = temp[i];
        }
        return ;
    }
}

通过对比归并排序和快速排序,我们可以发现,归并排序和快速排序的区别在于

//快速排序
int solder = partSort(array, left, right);
QuickSort(array, left, solder-1);
QuickSort(array, solder+1, right);
//归并排序
QuickSort(array, left, mid);
QuickSort(array, mid+1, right);
MergeArray(array,left,right,mid+1);

一个的QuickSort在两次递归之前,一个在两次递归之后,因为归并是先拆后再合并,而快速排序我们需要知道哨兵的位置,所以需要先进行局部排序找到哨兵。

文章来源:https://blog.csdn.net/qq_45816864/article/details/135176279
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。