热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

数据结构——快速排序的优化

文章目录1.快速排序的特性2.寻找基准的方法2.1Hoare法2.2挖坑法2.3前后指针法3.快速排序优化3.1划分不均匀3.2对后几层使用插入排序1.快速排序的特

文章目录

1.快速排序的特性

2.寻找基准的方法

2.1 Hoare法

2.2挖坑法

2.3前后指针法

3.快速排序优化

3.1划分不均匀

3.2 对后几层使用插入排序




1.快速排序的特性

时间复杂度:O(N*log(N))
空间复杂度:O(log(N))
稳定性:不稳定 当数据有序时,时间复杂度O(N^2)空间复杂度O(N)



2.寻找基准的方法


2.1 Hoare法

public static void quickSort(int[] array){sort(array,0, array.length-1);}private static void sort(int[] array,int start,int end){//防止只有左子树或者右子树的情况if(start >= end){return;}int povit = partion(array,start,end);sort(array,start,povit-1);sort(array,povit+1,end);}private static int partion(int[] array,int left,int right){//记录原始位置下标,方便后面和povit位置交换int i = left;//寻找参考值int povit = array[i];while(leftright,要加条件right--;}while(left = array[right]){left++;}//交换swap(array,left,right);}//交换povit到相遇的位置swap(array,left,i);return left;}


             * 问题1:为什么先从right 向左找?
             * 从右边开始找基准位置,是因为从左边开始找,左边先找到一个比array[left]大的数,然后右边
             * 向左找,左右肯定相遇,这时候交换,会把比基准值大的数交换到基准位置左边,达不到二分的目的了
             *
             * 问题2:为什么左右数组值比较时要带等于号?
             * 如果出现两边开始时值相同的情况,即左值不小于也不大于右值,两个循环都不会进去
             * 只会完成左右值一直交换,死循环了


2.2挖坑法

private static int paration1(int[] array,int left,int right){int povit = array[left];while(left= povit) {right--;}array[left] = array[right];while(left

2.3前后指针法

private static int paration2(int[] array,int left,int right){int prev = left;int cur = left+1;while(cur <= right){if(array[cur]

3.快速排序优化


3.1划分不均匀

当遇到单分支树的情况,会出现划分不均匀的问题,就会导致递归次数太多

三数取中法

private static void sort1(int[] array,int start,int end){//防止只有左子树或者右子树的情况if(start >= end){return;}//在找基准之前,解决划分不均匀的问题,将关键值改变为中间大小的值后,能解决单分支的情况int index = findMidValOfIndex(array,start,end);swap(array,start,index);int povit = partion(array,start,end);sort(array,start,povit-1);sort(array,povit+1,end);}/*找到中位数*/private static int findMidValOfIndex(int[] array,int start,int end){int midIndex = (start+end)/2;if(array[start] array[start]){return start;} else if (array[midIndex]

3.2 对后几层使用插入排序

当递归的区间很小的时候我们可以用插入排序,二叉树后几层节点数占总体节点数的大部分,
递归次数最多也发生在后几层,往后也越来越有序,就不递归了。用插入排序

private static void sort2(int[] array,int start,int end){//防止只有左子树或者右子树的情况if(start >= end){return;}if(( end-start+1) <= 15){//插入排序减少后几层 的递归insertSort1(array,start,end);}//在找基准之前,解决划分不均匀的问题,将关键值改变为中间大小的值后,能解决单分支的情况int index = findMidValOfIndex(array,start,end);swap(array,start,index);int povit = partion(array,start,end);sort(array,start,povit-1);sort(array,povit+1,end);}public static void insertSort1(int[] array,int left,int right){for (int i = left+1; i <= right ; i++) {int j = i-1;int tmp = array[i];for (; j >= left ; j--) {if(array[j] > array[i]){array[j+1] = array[j];}else{break;}}array[j+1] = tmp;}}


推荐阅读
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • Java 类成员初始化顺序与数组创建
    本文探讨了Java中类成员的初始化顺序、静态引入、可变参数以及finalize方法的应用。通过具体的代码示例,详细解释了这些概念及其在实际编程中的使用。 ... [详细]
  • 本文介绍如何在 Android 中通过代码模拟用户的点击和滑动操作,包括参数说明、事件生成及处理逻辑。详细解析了视图(View)对象、坐标偏移量以及不同类型的滑动方式。 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 本文详细介绍了Java中org.neo4j.helpers.collection.Iterators.single()方法的功能、使用场景及代码示例,帮助开发者更好地理解和应用该方法。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • Explore how Matterverse is redefining the metaverse experience, creating immersive and meaningful virtual environments that foster genuine connections and economic opportunities. ... [详细]
  • 本文介绍如何使用Objective-C结合dispatch库进行并发编程,以提高素数计数任务的效率。通过对比纯C代码与引入并发机制后的代码,展示dispatch库的强大功能。 ... [详细]
  • Java 中 Writer flush()方法,示例 ... [详细]
  • 本文介绍了如何使用 Spring Boot DevTools 实现应用程序在开发过程中自动重启。这一特性显著提高了开发效率,特别是在集成开发环境(IDE)中工作时,能够提供快速的反馈循环。默认情况下,DevTools 会监控类路径上的文件变化,并根据需要触发应用重启。 ... [详细]
  • Java 中的 BigDecimal pow()方法,示例 ... [详细]
  • 数据管理权威指南:《DAMA-DMBOK2 数据管理知识体系》
    本书提供了全面的数据管理职能、术语和最佳实践方法的标准行业解释,构建了数据管理的总体框架,为数据管理的发展奠定了坚实的理论基础。适合各类数据管理专业人士和相关领域的从业人员。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 在金融和会计领域,准确无误地填写票据和结算凭证至关重要。这些文件不仅是支付结算和现金收付的重要依据,还直接关系到交易的安全性和准确性。本文介绍了一种使用C语言实现小写金额转换为大写金额的方法,确保数据的标准化和规范化。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
author-avatar
mobiledu2502875393
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有