冒泡排序、选择排序、直接插入排序

冒泡排序

import java.util.Arrays;

/**
* @author dengqixing
* @date 2021/4/17
*/
public class BubbleSort {
public static void main(String[] args) {
// 1、定义无序数组
int[] arr = {89, 59, 44, 12, 58, 26, 94, 98, 21, 23};
// int[] arr = {12, 21, 23, 26, 44, 58, 59, 89, 94, 98}; // 2、输出无序数组
System.out.println(Arrays.toString(arr));
// 3、排序
// 外循环,至少比较数组长度减一,因为最后一个元素无需再进行比较
for (int i = 0; i < arr.length - 1; i++) {
// 默认序列是有序的
boolean flag = true;
// 内循环,由于每次比较都会有一个最大或最小的数被放在最边缘的位置,所以每次循环时都需要对应的减少比较的次数
for (int j = 0; j < arr.length - 1 - i; j++) {
if (arr[j] > arr[j + 1]) {
arr[j] = arr[j] ^ arr[j + 1];
arr[j + 1] = arr[j] ^ arr[j + 1];
arr[j] = arr[j] ^ arr[j + 1];
// 如果第一次小循环进行过交换,说明序列不是有序的,否则说明是有序的
flag = false;
}
}
// 这么做如果数组已经是有序了,则无需继续进行比较,减少循环次数,增加效率
if (flag) {
break;
}
} // 4、输出排序后的数组
System.out.println(Arrays.toString(arr));
}
}

选择排序

class SelectSort {
public static void main(String[] args) {
// 1、定义无序数组
int[] arr = {89, 59, 44, 12, 58, 26, 94, 98, 21, 23}; // 2、输出无序数组
System.out.println(Arrays.toString(arr));
// 3、排序
// 外循环,需要 n - 1的循环次数
for (int i = 0; i < arr.length - 1; i++) {
int minIndex = i;
for (int j = i + 1; j < arr.length; j++) {
if (arr[minIndex] > arr[j]) {
minIndex = j;
}
}
// 进行交换
if (minIndex != i) {
int temp = arr[minIndex];
arr[minIndex] = arr[i];
arr[i] = temp;
}
} // 4、输出排序后的数组
System.out.println(Arrays.toString(arr));
}
}

选择排序与冒泡排序的注意事项

  • 冒泡排序至多需要n-1趟循环,最少一趟循环;其中n为长度
  • 选择排序一定需要n-1躺循环,一趟也不能少
  • 冒泡排序中最多的操作就是比较和交换,一趟循环中可能发生多次交换;
  • 选择排序中最多的操作就是比较,一趟比较结束后发现更小(或更大)的值才进行交换
  • 所以更推荐使用冒泡。

小案例,使用选择排序完成对对象的排序

class SelectSort2 {
public static void main(String[] args) {
// 1、定义无序数组
Comparable[] arr = {new Person(14,"李四"),new Person(18,"李四"),new Person(16,"李四")}; // 2、输出无序数组
System.out.println(Arrays.toString(arr));
// 3、排序
selectSort(arr); // 4、输出排序后的数组
System.out.println(Arrays.toString(arr));
} private static void selectSort(Comparable[] arr) {
for (int i = 0; i < arr.length - 1; i++) {
int minIndex = i;
for (int j = i + 1; j < arr.length; j++) {
if (arr[minIndex].compareTo(arr[j]) > 0) {
minIndex = j;
}
}
// 进行交换
if (minIndex != i) {
Comparable temp = arr[minIndex];
arr[minIndex] = arr[i];
arr[i] = temp;
}
}
}
} class Person implements Comparable<Person> {
int age;
String name; public Person(int age, String name) {
this.age = age;
this.name = name;
} public int getAge() {
return age;
} public void setAge(int age) {
this.age = age;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} @Override
public int compareTo(Person p) {
return Integer.compare(this.getAge(), p.getAge());
} @Override
public String toString() {
return "Person{" +
"age=" + age +
", name='" + name + '\'' +
'}';
}
}

直接插入排序(插入排序)

public static void insertSort(int[] arr) {
// 默认第一个元素是有序序列,所以从1开始
for (int i = 1; i < arr.length; i++) {
// 临时变量存储当前需要插入的数据
int temp = arr[i];
// 存储有序序列的元素下标位置
int j = i - 1;
// 如果当前的元素小于需要比较的元素
while (j >= 0 && temp < arr[j]) {
// 将当前正在比较的元素放入其后面的位置
arr[j + 1] = arr[j];
// 继续往左进行数据比较
j--;
}
// 循环结束后,将其比较的元素放在最后比较的一个元素的后面
arr[j + 1] = temp;
}
}

快速排序(比较排序中效率最高的一种排序)

原理:

  • 首先得到一个无序序列
  • 随机选取其中一个数为基准数(一般默认为最左侧的第一个元素)
  • 然后会有两个指针分别在当前需要比较的序列的最左侧和最右侧
  • 先从最右侧开始往左移动比较,一旦碰到了大于或者相等于基准数的元素,则继续向左移动,直到指针所指向的元素小于基准数,就停下(还有一个条件,左边的索引必须小于右边的索引)
  • 然后从最左侧开始向右移动比较,一旦碰到了小于或者相等于基准数的元素,则继续向右移动,直到指针所指向的元素大于基准数,就停下(左边的索引必须小于右边的索引)
  • 如果左指针与右指针相等(也就是相遇了),则将其相遇位置的元素与基准数进行交换
  • 最后再将基准数左边的序列再次进行快速排序
  • 将基准数右边的序列再次进行快速排序
public static void main(String[] args) {
int[] arr = new int[100000000];
for (int i = 0; i < arr.length; i++) {
// 随机存入1到10万之间的数字
arr[i] = (int)(Math.random() * 100000)+1;
} // 排序
long start = System.currentTimeMillis();
quickSort(arr, 0 , arr.length -1);
long end = System.currentTimeMillis();
//19357毫秒,排序1亿个数字的序列竟只用了19秒,
冒泡排序在10万数据时就已耗时15秒左右,快速排序在10万数据时耗时只有20-30毫秒
System.out.println("运行时间为 :" + (end - start));
} public static void quickSort(int[] arr, int left, int right) {
// 如果左边索引比右边索引还要大,是不合法的,直接return结束方法
if (left > right) {
return;
}
// 获得基准数(默认选择最左边的数)与左右2个下标索引数
int base = arr[left];
int i = left;
int j = right; // 当i和j不相遇的时候,才进行循环检索
while (i != j) {
// 从右边开始检索,如果找到了小于基准数的数,就停下
// 如果检索到比基准数大的或者相等的,就继续检索
while (arr[j] >= base && i < j) {
j--;
}
// 从左开始检索,如果找到了大于基准数的数,就停下
// 如果检索到比基准数小的或者相等的,就继续检索
while (arr[i] <= base && i < j) {
i++;
}
// 到这说明i和j都停下了,交换它们两的元素
int temp = arr[i];
arr[i] = arr[j];
arr[j] = temp;
}
// 到这说明 i 和 j相遇了,就将相遇位置的值与基准数交换
arr[left] = arr[i];
arr[i] = base; // 然后将排序后左边的序列再次进行排序
quickSort(arr, left , i -1);
// 将右边的序列再次进行排序
quickSort(arr, i + 1, right);
}
}

折半查找(使用时有限制,只能是排序好了的数组)

// 非递归二分查找
public static int binarySearch(int[] arr, int key) {
int low = 0;
int high = arr.length - 1;
while (low <= high) {
int mid = (low + high) / 2;
if (arr[mid] == key) {
return mid;
} else if (key < arr[mid]) {
high = mid - 1;
} else if (key > arr[mid]) {
low = mid + 1;
}
}
return -1;
} // 递归二分查找
public static int binarySearch2(int[] arr, int key) {
int low = 0;
int high = arr.length - 1;
int index = binarySearch(arr, key, low, high);
return index;
} private static int binarySearch(int[] arr, int key, int low, int high) {
if (low > high) {
return -1;
}
// 计算中间索引
int mid = (low + high) / 2;
if (arr[mid] == key) {
return mid;
} else if (key < arr[mid]) {
return binarySearch(arr, key, low, mid - 1);
} else {
return binarySearch(arr, key, mid + 1, high);
}
}

补充一下递归的优点与缺点

  • 优点:编码简单,适合处理相同业务逻辑的问题
  • 缺点:占用内存空间多,每递归一次都会在栈空间中开辟新空间执行方法

冒泡排序、选择排序、直接插入排序、快速排序、折半查找>从零开始学JAVA系列的更多相关文章

  1. 09-java学习-数组-冒泡排序-选择排序-数组工具类编写-查找-扩容

    数组的排序算法 查找算法 数组协助类Arrays的学习和使用 数组的扩容

  2. Java排序算法分析与实现:快排、冒泡排序、选择排序、插入排序、归并排序(二)

    一.概述: 上篇博客介绍了常见简单算法:冒泡排序.选择排序和插入排序.本文介绍高级排序算法:快速排序和归并排序.在开始介绍算法之前,首先介绍高级算法所需要的基础知识:划分.递归,并顺带介绍二分查找算法 ...

  3. Java数据结构和算法总结-冒泡排序、选择排序、插入排序算法分析

    前言:排序在算法中的地位自然不必多说,在许多工作中都用到了排序,就像学生成绩统计名次.商城商品销量排名.新闻的搜索热度排名等等.也正因为排序的应用范围如此之广,引起了许多人深入研究它的兴趣,直至今天, ...

  4. 算法 排序lowB三人组 冒泡排序 选择排序 插入排序

    参考博客:基于python的七种经典排序算法   [经典排序算法][集锦]     经典排序算法及python实现 首先明确,算法的实质 是 列表排序.具体就是操作的列表,将无序列表变成有序列表! 一 ...

  5. JavaScript算法(冒泡排序、选择排序与插入排序)

    冒泡排序.选择排序与插入排序复杂度都是二次方级别的,放在一起说吧. 介绍一些学习这三个排序方法的比较好的资料.冒泡排序看<学习JavaScript数据结构与算法>介绍的冒泡排序,选择排序看 ...

  6. Java选择排序,插入排序,快速排序

      public class Test { public static void main(String[] args) { int a[] = { 1, 2, 3, 4, 5 }; 选择排序(a); ...

  7. 学习C#之旅 冒泡排序,选择排序,插入排序,希尔排序[资料收集]

    关于冒泡排序,选择排序,插入排序,希尔排序[资料收集]  以下资料来源与网络 冒泡排序:从后到前(或者从前到后)相邻的两个两两进行比较,不满足要求就位置进行交换,一轮下来选择出一个最小(或最大)的放到 ...

  8. C# 插入排序 冒泡排序 选择排序 高速排序 堆排序 归并排序 基数排序 希尔排序

    C# 插入排序 冒泡排序 选择排序 高速排序 堆排序 归并排序 基数排序 希尔排序 以下列出了数据结构与算法的八种基本排序:插入排序 冒泡排序 选择排序 高速排序 堆排序 归并排序 基数排序 希尔排序 ...

  9. python算法(一)基本知识&冒泡排序&选择排序&插入排序

    本节内容: 算法基本知识 冒泡排序 选择排序 插入排序 1. 算法基本知识 1.1 什么是算法? 算法(algorithm):就是定义良好的计算过程,他取一个或一组的值为输入,并产生出一个或一组值作为 ...

随机推荐

  1. Java-学习日记(Java8异步)

    今天用到的中异步操作:异步编程与异步处理数据 //里面返回其他接口服务使用CompletableFuture CompletableFuture.runAsync(()->{ driverNoR ...

  2. Ubuntu安装部署Kafka

    Ubuntu安装部署Kafka 环境: Ubuntu 18.04.4 LTS ,JDK1.8,kafka_2.12-2.3.1 确保已经安装了JDK,JDK安装过程不再赘述.可参考文章xxxx 一.下 ...

  3. (Element UI 组件 Table)去除单元格底部的横线

    Element UI 组件 Table 有一个属性 border,添加它可以增加纵向边框,但是无法控制横线边框,因此即使是最简单的 el-table,也会包含一个底部横线. 这个底部横线其实是一个 b ...

  4. Vue3全家桶升级指南一composition API

    1.setup() vue3中的composition API中最重要的就是setup方法了,相当于组件的入口,所有的composition API都必须放到setup()中的使用. setup是在组 ...

  5. 如何获取微信小程序for循环的index

    在微信小程序开发中,对于wx:for,可以使用wx:for-index="index"来获取数组中的元素的索引值(下标). <view class="item&qu ...

  6. webpack(8)vue组件化开发的演变过程

    前言 真实项目开发过程中,我们都是使用组件化的去开发vue的项目,但是组件化的思想又是如何来的呢?下面就从开始讲解演变过程 演变过程1.0 一般情况下vue都是单页面开发,所以项目中只会有一个inde ...

  7. kali安装angr

    最近打算重新学习一波angr,先把环境搭好 1. 先安装virtualenv,这玩意是可以创建一个纯净的python环境,我陷入了沉思,pyenv好像也可以 这里利用豆瓣的源下载,非常快而且很舒服 p ...

  8. mybatis 配置的log4j文件无效,不能正常显示日志信息

    正在学习mybatis,配置好后log4j.properties文件后,日志信息不能正常显示,没有效果. 查看了一下mybatis的相关文档,在日志一栏找到问题愿意 原因是我们的mybatis选了其他 ...

  9. Maven | 把jar包安装到本地仓库

    使用的场景 自己写的工具类想安装到本地 从Maven仓库中下载不下来的jar 使用的步骤 首先要保证自己的Maven配置全局环境变量,如果没有配置过maven全局变量,可以按照下面的步骤配置一下: 先 ...

  10. dp 套 dp扯谈

    1.[扯谈概念] \(dp\) 套 \(dp\) 其实也就是 \(dp\) . 这里就定义下面两个概念: 内层 \(dp\) 表示的是被套在里面的那个 \(dp\) 外层 \(dp\) 表示的是最外面 ...