当前位置: 首页 > news >正文

冒泡排序与快速排序


 博主主页: 码农派大星.

    数据结构专栏:Java数据结构

 数据库专栏:MySQL数据库

关注博主带你了解更多数据结构知识


1.冒泡排序

冒泡排序

private static void swap(int[] arrary,int i,int j){int tmp = arrary[i];arrary[i] = arrary[j];arrary[j] = tmp;public static void  bubbleSort(int[] arrary){for (int i = 0; i <arrary.length-1 ; i++) {for (int j = 0; j < arrary.length-1-i; j++) {if(arrary[j]> arrary[j+1]){swap(arrary,j,j+1);}}}return arrary;}

冒泡排序总结

1. 冒泡排序是一种非常容易理解的排序

2. 时间复杂度:O(N^2)

3. 空间复杂度:O(1)

4. 稳定性:稳定 

2.快速排序

快速排序是Hoare于1962年提出的一种二叉树结构的交换排序方法,其基本思想为:任取待排序元素序列中的某元 素作为基准值,按照该排序码将待排序集合分割成两子序列,左子序列中所有元素均小于基准值,右子序列中所有 元素均大于基准值,然后最左右子序列重复该过程,直到所有元素都排列在相应位置上为止。

1.Hoare版

public static void  quickSort(int[] arrary){quick(arrary,0,arrary.length-1);return arrary;}private static void swap(int[] arrary,int i,int j){int tmp = arrary[i];arrary[i] = arrary[j];arrary[j] = tmp;private static void quick(int [] arrary,int start,int end){if (start >= end) {return;}int par = partition(arrary,start,end);quick(arrary,start,par-1);quick(arrary,par+1,end);}private static int partition(int [] arrary,int left,int right){int i = left;int tmp = arrary[left];while (left < right){//right-- : 先走左边会导致最后相遇的地方比基准大的数据,// 交换完后,会把大于基准的值换到前面while (left < right && arrary[right] >= tmp){right--;}while (left < right && arrary[left] <= tmp){left++;}swap(arrary,left,right);}//此时相遇left=right;swap(arrary,left,i);return right;}

2.挖坑法 

public static void  quickSort(int[] arrary){quick(arrary,0,arrary.length-1);return arrary;}
private static void quick(int [] arrary,int start,int end){if (start >= end) {return;}int par = partitionWaken(arrary,start,end);quick(arrary,start,par-1);quick(arrary,par+1,end);}
private static int partitionWaken(int [] arrary,int left,int right){int tmp = arrary[left];while (left<right){while (left < right && arrary[right] >= tmp){right--;}arrary[left] = arrary [right];while (left<right && arrary[left] <= tmp){left++;}arrary[right] = arrary[left];}arrary[left] = tmp;return left;}

3.快速排序优化

1. 三数取中法选key

public static void  quickSort(int[] arrary){quick(arrary,0,arrary.length-1);
return arrary;}private static void quick(int [] arrary,int start,int end){if (start >= end) {return;}int index = midThreeNum(arrary,start,end);swap(arrary,index,start);int par = partitionWaken(arrary,start,end);quick(arrary,start,par-1);quick(arrary,par+1,end);}private static int partitionWaken(int [] arrary,int left,int right){int tmp = arrary[left];while (left<right){while (left < right && arrary[right] >= tmp){right--;}arrary[left] = arrary [right];while (left<right && arrary[left] <= tmp){left++;}arrary[right] = arrary[left];}arrary[left] = tmp;return left;}private static int midThreeNum(int [] arrary,int left,int right){int mid = (left+right)/2;if (arrary[left] < arrary[right]){if (arrary[mid] < arrary[left]) {return left;}else if (arrary[mid] > arrary[right]){return right;}else {return mid;}}else{if (arrary[mid] < arrary[right]){return right;}else if(arrary[mid] > arrary[left]){return left;}else {return mid;}}}

2. 递归到小的子区间时,可以考虑使用插入排序

我们在数组中数据小于等于10时改为插入排序,提高了排序的效率.

 public static void  quickSort(int[] arrary){quick(arrary,0,arrary.length-1);
return arrary;}private static void quick(int [] arrary,int start,int end){if (start >= end) {return;}if (end - start + 1 <= 10) {inserSortRange(arrary,start,end);return;}int index = midThreeNum(arrary,start,end);swap(arrary,index,start);int par = partitionWaken(arrary,start,end);quick(arrary,start,par-1);quick(arrary,par+1,end);}public static  void inserSortRange(int [] array,int left,int right){for(int i = left+1; i< right;i++){int tmp = array[i];int j = i-1;for (; j >=0 ; j--) {if (array[j] > tmp) {array[j+1] = array[j];}else {//array[j+1]= tmp;break;}}array[j+1]= tmp;}}private static int partitionWaken(int [] arrary,int left,int right){int tmp = arrary[left];while (left<right){while (left < right && arrary[right] >= tmp){right--;}arrary[left] = arrary [right];while (left<right && arrary[left] <= tmp){left++;}arrary[right] = arrary[left];}arrary[left] = tmp;return left;}private static int midThreeNum(int [] arrary,int left,int right){int mid = (left+right)/2;if (arrary[left] < arrary[right]){if (arrary[mid] < arrary[left]) {return left;}else if (arrary[mid] > arrary[right]){return right;}else {return mid;}}else{if (arrary[mid] < arrary[right]){return right;}else if(arrary[mid] > arrary[left]){return left;}else {return mid;}}}

4.非递归的快速排序 

//非递归快速排序public static void quickNot(int[] array){Stack<Integer> stack = new Stack<>();int left = 0;int right = array.length - 1;int par = partition(array,left,right);if (par > left+1){stack.push(left);stack.push(par-1);}if (par < right-1){stack.push(par+1);stack.push(right);}while (!stack.isEmpty()){right = stack.pop();left = stack.pop();par = partitionWaken(array,left,right);if(par > left+1){stack.push(left);stack.push(par-1);}if (par < right -1){stack.push(par+1);stack.push(right);}}return array;}
private static int partition(int [] arrary,int left,int right){int i = left;int tmp = arrary[left];while (left < right){//right-- : 先走左边会导致最后相遇的地方比基准大的数据,// 交换完后,会把大于基准的值换到前面while (left < right && arrary[right] >= tmp){right--;}while (left < right && arrary[left] <= tmp){left++;}swap(arrary,left,right);}//此时相遇left=right;swap(arrary,left,i);return right;}

未优化的快速排序,再遇到数据过多时,程序会崩. 

1. 快速排序整体的综合性能和使用场景都是比较好的,所以才敢叫快速排序

2. 时间复杂度:O(N*logN)

快速排序和堆排序时间复杂度一样,但是快速排序要比堆排序快

3. 空间复杂度:O(logN)

4. 稳定性:不稳定

相关文章:

  • C#中的值类型与引用类型
  • 计算机毕业设计 | SpringBoot+vue仓库管理系统(附源码)
  • 欧科云链:Web3.0时代 具备链上数据分析能力的公司愈发凸显其价值
  • JCR一区级 | Matlab实现TCN-BiGRU-MATT时间卷积双向门控循环单元多特征分类预测
  • 解释Python中的PEP 8是什么 为什么它很重要
  • 基于Chisel的FPGA流水灯设计
  • ios:文本框默认的copy、past改成中文复制粘贴
  • 平移数据c++
  • 【吊打面试官系列】Java高并发篇 - 什么是自旋 ?
  • js实现基础购物车的制作
  • Debian常用指令指南:高效管理你的Linux系统
  • vue-标签选择
  • HTML (总结黑马的)
  • JVM学习笔记(持续更新)
  • React(四)memo、useCallback、useMemo Hook
  • 「前端」从UglifyJSPlugin强制开启css压缩探究webpack插件运行机制
  • 【附node操作实例】redis简明入门系列—字符串类型
  • 78. Subsets
  • Asm.js的简单介绍
  • CSS进阶篇--用CSS开启硬件加速来提高网站性能
  • gops —— Go 程序诊断分析工具
  • JDK 6和JDK 7中的substring()方法
  • JS+CSS实现数字滚动
  • 当SetTimeout遇到了字符串
  • 搞机器学习要哪些技能
  • 记一次删除Git记录中的大文件的过程
  • 前端临床手札——文件上传
  • 前嗅ForeSpider教程:创建模板
  • 区块链共识机制优缺点对比都是什么
  • 如何用Ubuntu和Xen来设置Kubernetes?
  • 如何正确配置 Ubuntu 14.04 服务器?
  • 首页查询功能的一次实现过程
  • 学习笔记:对象,原型和继承(1)
  • hi-nginx-1.3.4编译安装
  • Unity3D - 异步加载游戏场景与异步加载游戏资源进度条 ...
  • 基于django的视频点播网站开发-step3-注册登录功能 ...
  • #[Composer学习笔记]Part1:安装composer并通过composer创建一个项目
  • (52)只出现一次的数字III
  • (day 2)JavaScript学习笔记(基础之变量、常量和注释)
  • (Pytorch框架)神经网络输出维度调试,做出我们自己的网络来!!(详细教程~)
  • (动手学习深度学习)第13章 计算机视觉---微调
  • (独孤九剑)--文件系统
  • (分布式缓存)Redis持久化
  • (附源码)ssm基于web技术的医务志愿者管理系统 毕业设计 100910
  • (附源码)ssm经济信息门户网站 毕业设计 141634
  • (力扣记录)1448. 统计二叉树中好节点的数目
  • (十一)图像的罗伯特梯度锐化
  • (新)网络工程师考点串讲与真题详解
  • (原創) X61用戶,小心你的上蓋!! (NB) (ThinkPad) (X61)
  • (转)ABI是什么
  • (转)Oracle 9i 数据库设计指引全集(1)
  • (转)菜鸟学数据库(三)——存储过程
  • *算法训练(leetcode)第四十七天 | 并查集理论基础、107. 寻找存在的路径
  • ... fatal error LINK1120:1个无法解析的外部命令 的解决办法
  • .NET C# 使用 iText 生成PDF