Java 排序算法-冒泡排序及其优化
什么是冒泡排序
这里引用一下百度百科上的定义:
冒泡排序(Bubble Sort),是一种计算机科学领域的较简单的排序算法。
它重复地走访过要排序的元素列,依次比较两个相邻的元素,如果顺序(如从大到小、首字母从Z到A)错误就把他们交换过来。走访元素的工作是重复地进行直到没有相邻元素需要交换,也就是说该元素列已经排序完成。
这个算法的名字由来是因为越小的元素会经由交换慢慢“浮”到数列的顶端(升序或降序排列),就如同碳酸饮料中二氧化碳的气泡最终会上浮到顶端一样,故名“冒泡排序”。
冒泡排序在实际工作中用的并不多,多是面试题中会遇到。
下面用java代码来实现:
基本写法
为了看到后面优化后的效果,我加了count 和 num l两个变量来分布统计外循环和内循环的次数
/**
* 原始的冒泡排序
* @param values
*/
public static void bubbleSort(int[] values) {
System.out.println("排序前:" + Arrays.toString(values));
int tmp;// 循环外创建变量,减少垃圾产生,提高性能
int count = 0;// 比较了多少趟
int num = 0; // 比较次数
// 外层循环,n 个元素,最多需要 n -1 趟循环
for(int i = 0; i < values.length -1 ; i++) {
count ++;
// 内层循环, 每一次都从开始两个元素开始两两比较到最后
for (int j = 0; j < values.length - 1; j++) {
if(values[j] > values[j + 1]) {
tmp = values[j];
values[j] = values[j + 1];
values[j + 1] = tmp;
}
num ++;
}
}
System.out.println("排序后:" + Arrays.toString(values));
System.out.println("比较趟数:"+count + ",比较次数:" + num);
}
优化后写法
/**
* 优化了比较次数的冒泡排序
* @param values
*/
public static void bubbleSort2(int[] values) {
System.out.println("排序前:" + Arrays.toString(values));
int tmp; // 循环外创建变量,减少垃圾产生,提高性能
int count = 0;// 比较了多少趟
int num = 0; // 比较次数
// 外层循环,n 个元素,最多需要 n -1 趟循环
for(int i = 0; i < values.length -1 ; i++) {
count ++;
// 内层循环, 每一次都从开始两个元素开始两两比较到最后
/* 每一趟循环都可以找你最大的数,内循环可以比上一趟循环减少一次比较 */
for (int j = 0; j < values.length - 1 - i; j++) {
if(values[j] > values[j + 1]) {
tmp = values[j];
values[j] = values[j + 1];
values[j + 1] = tmp;
}
num ++;
}
}
System.out.println("排序后:" + Arrays.toString(values));
System.out.println("比较趟数:"+count + ",比较次数:" + num);
}
显然,每一趟循环都可以找你最大的数,内循环可以比上一趟循环减少一次,面试时写成这样基本OK了。
终极版本
实际中,用来比较的数据,可能就只有几个位置是无序的,如果也按照上面的写法进行 length -1 次循环就显得有点死板了。
因此我们可以假设数组已经排好序了,给程序一个标记变量,如果标记变量显示数组已排好序,则可以跳出循环。
/**
* 冒泡排序终极写法
* @param values
*/
public static void bubbleSort3(int[] values) {
System.out.println("排序前:" + Arrays.toString(values));
int tmp;// 循环外创建变量,减少垃圾产生,提高性能
int count = 0;// 比较了多少趟
int num = 0; // 比较次数
// 外层循环,n 个元素,最多需要 n -1 趟循环
for(int i = 0; i < values.length -1 ; i++) {
// 定义一个boolean 值,标记数组是否达到有序状态
boolean flag = true;
count ++;
// 内层循环, 每一次都从开始两个元素开始两两比较到最后
/* 每一趟循环都可以找你最大的数,内循环可以比上一趟循环减少一次比较 */
for (int j = 0; j < values.length - 1 - i; j++) {
if(values[j] > values[j + 1]) {
tmp = values[j];
values[j] = values[j + 1];
values[j + 1] = tmp;
// 如果本趟发生了交换,表明数组还是无序,需要进入下一轮循环
flag = false;
}
num ++;
}
// 根据标记的布尔值判断数组是否达到有序状态,如有序,则退出循环
if( flag) {
break;
}
}
System.out.println("排序后:" + Arrays.toString(values));
System.out.println("比较趟数:"+count + ",比较次数:" + num);
}
源码及测试
package com.xzlf.sort;
import java.util.Arrays;
public class TestSort {
public static void main(String[] args) {
int[] arr = {1,341,33,22,88,99,100};
int[] arr2 = {1,341,33,22,88,99,100};
int[] arr3 = {1,341,33,22,88,99,100};
bubbleSort(arr);
System.out.println("==================");
bubbleSort2(arr2);
System.out.println("==================");
bubbleSort3(arr3);
}
/**
* 原始的冒泡排序
* @param values
*/
public static void bubbleSort(int[] values) {
System.out.println("排序前:" + Arrays.toString(values));
int tmp;// 循环外创建变量,减少垃圾产生,提高性能
int count = 0;// 比较了多少趟
int num = 0; // 比较次数
// 外层循环,n 个元素,最多需要 n -1 趟循环
for(int i = 0; i < values.length -1 ; i++) {
count ++;
// 内层循环, 每一次都从开始两个元素开始两两比较到最后
for (int j = 0; j < values.length - 1; j++) {
if(values[j] > values[j + 1]) {
tmp = values[j];
values[j] = values[j + 1];
values[j + 1] = tmp;
}
num ++;
}
}
System.out.println("排序后:" + Arrays.toString(values));
System.out.println("比较趟数:"+count + ",比较次数:" + num);
}
/**
* 优化了比较次数的冒泡排序
* @param values
*/
public static void bubbleSort2(int[] values) {
System.out.println("排序前:" + Arrays.toString(values));
int tmp; // 循环外创建变量,减少垃圾产生,提高性能
int count = 0;// 比较了多少趟
int num = 0; // 比较次数
// 外层循环,n 个元素,最多需要 n -1 趟循环
for(int i = 0; i < values.length -1 ; i++) {
count ++;
// 内层循环, 每一次都从开始两个元素开始两两比较到最后
/* 每一趟循环都可以找你最大的数,内循环可以比上一趟循环减少一次比较 */
for (int j = 0; j < values.length - 1 - i; j++) {
if(values[j] > values[j + 1]) {
tmp = values[j];
values[j] = values[j + 1];
values[j + 1] = tmp;
}
num ++;
}
}
System.out.println("排序后:" + Arrays.toString(values));
System.out.println("比较趟数:"+count + ",比较次数:" + num);
}
/**
* 冒泡排序终极写法
* @param values
*/
public static void bubbleSort3(int[] values) {
System.out.println("排序前:" + Arrays.toString(values));
int tmp;// 循环外创建变量,减少垃圾产生,提高性能
int count = 0;// 比较了多少趟
int num = 0; // 比较次数
// 外层循环,n 个元素,最多需要 n -1 趟循环
for(int i = 0; i < values.length -1 ; i++) {
// 定义一个boolean 值,标记数组是否达到有序状态
boolean flag = true;
count ++;
// 内层循环, 每一次都从开始两个元素开始两两比较到最后
/* 每一趟循环都可以找你最大的数,内循环可以比上一趟循环减少一次比较 */
for (int j = 0; j < values.length - 1 - i; j++) {
if(values[j] > values[j + 1]) {
tmp = values[j];
values[j] = values[j + 1];
values[j + 1] = tmp;
// 如果本趟发生了交换,表明数组还是无序,需要进入下一轮循环
flag = false;
}
num ++;
}
// 根据标记的布尔值判断数组是否达到有序状态,如有序,则退出循环
if( flag) {
break;
}
}
System.out.println("排序后:" + Arrays.toString(values));
System.out.println("比较趟数:"+count + ",比较次数:" + num);
}
}
上面代码运行结果:

@[TOC](Java/java基础)
Java 排序算法-冒泡排序及其优化的更多相关文章
- Java排序算法——冒泡排序
import java.util.Arrays; //================================================= // File Name : Bubble_S ...
- 常用Java排序算法
常用Java排序算法 冒泡排序 .选择排序.快速排序 package com.javaee.corejava; public class DataSort { public DataSort() { ...
- java排序算法(四):冒泡排序
java排序算法(四):冒泡排序 冒泡排序是计算机的一种排序方法,它的时间复杂度是o(n^2),虽然不及堆排序.快速排序o(nlogn,底数为2).但是有两个优点 1.编程复杂度很低.很容易写出代码 ...
- java排序算法之冒泡排序(Bubble Sort)
java排序算法之冒泡排序(Bubble Sort) 原理:比较两个相邻的元素,将值大的元素交换至右端. 思路:依次比较相邻的两个数,将小数放在前面,大数放在后面.即在第一趟:首先比较第1个和第2个数 ...
- java排序算法之冒泡排序和快速排序
总结一下Java排序算法,以便记忆. 各类排序的时间复杂度: 排序方法 时间复杂度(平均) 时间复杂度(最坏) 时间复杂度(最好) 空间复杂度 稳定性 复杂性 直接插入排序 O(n2)O(n2) O( ...
- java排序算法(一):概述
java排序算法(一)概述 排序是程序开发中一种非常常见的操作,对一组任意的数据元素(活记录)经过排序操作后,就可以把它们变成一组按关键字排序的一组有序序列 对一个排序的算法来说,一般从下面三个方面来 ...
- java排序算法(六):直接插入排序
java排序算法(六):直接插入排序 直接插入排序的基本操作就是将待的数据元素按其关键字的大小插入到前面的有序序列中 直接插入排序时间效率并不高,如果在最坏的情况下,所有元素的比较次数的总和为(0+1 ...
- Java排序算法之快速排序
Java排序算法之快速排序 快速排序(Quicksort)是对冒泡排序的一种改进. 快速排序由C. A. R. Hoare在1962年提出.它的基本思想是:通过一趟排序将要排序的数据分割成独立的两部分 ...
- Java排序算法(三)
Java排序算法(三) 三.Java排序算法总结 从这三组时间复杂度对比中,可以看出,堆排序和归并排序是不管在什么情况下发挥稳定的,快速排序好的时候表现如天才,坏情况下比较差强人意,甚至在等待排序个数 ...
随机推荐
- Mayor's posters POJ - 2528 线段树(离散化处理大数?)
题意:输入t组数据,输入n代表有n块广告牌,按照顺序贴上去,输入左边和右边到达的地方,问贴完以后还有多少块广告牌可以看到(因为有的被完全覆盖了). 思路:很明显就是线段树更改区间,不过这个区间的跨度有 ...
- CodeForces - 1249E 楼梯和电梯
题意:第一行输入n和c,表示有n层楼,电梯来到需要时间c 输入两行数,每行n-1个,表示从一楼到二楼,二楼到三楼.....n-1楼到n楼,a[ ] 走楼梯和 b[ ] 乘电梯花费的时间 思路:动态规划 ...
- 83 项开源视觉 SLAM 方案够你用了吗?
作者:吴艳敏 来源:83 项开源视觉 SLAM 方案够你用了吗? 前言 1. 本文由知乎作者小吴同学同步发布于https://zhuanlan.zhihu.com/p/115599978/并持续更新. ...
- arcgis连接oracle发布服务,提示数据未注册到服务器,手动注册服务器失败
arcgis连接oracle数据库发布服务时候,分析之后提示:数据未注册到服务器上. 手动注册之后提示:数据客户端没有正确配置.实际上数据库客户端已经安装完成也可以使用. 设置 PATH 环境变量(仅 ...
- ovirt 替换自主签署证书
需求我自己写了一个python后台,添加上了ovirt 引擎web上,如图 但第一次访问时需要,需要接受两次不安全连接,ovirt web使用https,我往里面加http,加不进去. 只能同样使用 ...
- python学习第四节 迭代器 生成器
1:什么是迭代 可以直接作用于for循环的对象统称为可迭代对象(Iterable). 可以被next()函数调用并不断返回下一个值的对象称为迭代器(Iterator). 所有的Iterable均可以通 ...
- web安全学习神器——DVWA安装部署
前言 DVWA是用PHP+Mysql编写的一套用于常规WEB漏洞教学和检测的WEB脆弱性测试程序.包含了SQL注入.XSS.文件包含等常见的一些安全漏洞.接下来我会用图文的形式讲解一下DVWA的下载与 ...
- .Net微服务实践(四)[网关]:Ocelot限流熔断、缓存以及负载均衡
目录 限流 熔断 缓存 Header转化 HTTP方法转换 负载均衡 注入/重写中间件 后台管理 最后 在上篇.Net微服务实践(三)[网关]:Ocelot配置路由和请求聚合中我们介绍了Ocelot的 ...
- shell使用特殊变量
shell使用特殊变量 3.1问题 本例要求编写一个脚本/root/myuse ...
- es--es分词的一些分析技巧
查看某个字段的分词结果 POST /index/tyhpe/id/_termvectors?fields=fields_name 例如:http://localhost:9200/prod_membe ...