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

Java面试排序算法

一、排序简介排序算法大体可分为两种:1、比较排序,时间复杂度O(nlogn)~O(n^2),主要有




一、排序简介
排序算法大体可分为两种:
1、比较排序,时间复杂度O(nlogn) ~ O(n^2),主要有:冒泡排序,选择排序,插入排序,归并排序,堆排序,快速排序等。
2、非比较排序,时间复杂度可以达到O(n),主要有:计数排序,基数排序,桶排序等。
这里写图片描述
二、冒泡排序法
算法思路:
1、比较相邻的元素。如果第一个比第二个大,就交换它们两个;
2、对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对,这样在最后的元素应该会是最大的数;
3、针对所有的元素重复以上的步骤,除了最后一个;
4、重复步骤1~3,直到排序完成。
这里写图片描述
代码:

public class
{
public static void main(String[] args) {
int array[] = {1,2,4,3,9,7,8,6};
for( int i &#61; 0;i <array.length - 1;i&#43;&#43; ){
for( int j &#61; 0;j <array.length - i - 1;j&#43;&#43; ){
if( array[j] > array[j&#43;1] ){
int temp &#61; array[j];
array[j] &#61; array[j&#43;1];
array[j&#43;1] &#61; temp;
}
}
}
for( int i &#61; 0 ; i <array.length ; i&#43;&#43; ){
System.out.print(array[i]&#43;" ");
}
}
}
//运行结果:
//1 2 3 4 6 7 8 9

三、选择排序
算法思路&#xff1a;
首先在未排序序列中找到最小&#xff08;大&#xff09;元素&#xff0c;存放到排序序列的起始位置&#xff0c;然后&#xff0c;再从剩余未排序元素中继续寻找最小&#xff08;大&#xff09;元素&#xff0c;然后放到已排序序列的末尾。以此类推&#xff0c;直到所有元素均排序完毕。
这里写图片描述
代码&#xff1a;

public class XuanZePaiXu {
public static void main(String[] args) {
int minIndex &#61; 0;
int temp &#61; 0;
int array[] &#61; {1,2,4,3,9,7,8,6};
for(int i &#61; 0;i <array.length;i&#43;&#43;){
minIndex &#61; i; //先假设最开始的元素为最小的元素
for( int j &#61; i &#43; 1;j <array.length;j&#43;&#43; ){
if( array[j] <array[minIndex] ){ // 寻找最小的数
minIndex &#61; j; // 将最小数的索引保存
}
}
temp &#61; array[minIndex]; //将此轮的最小元素和最开始的元素交换
array[minIndex] &#61; array[i];
array[i] &#61; temp;
}
for( int i &#61; 0;i <array.length;i&#43;&#43; ){
System.out.print(array[i]&#43;" ");
}
}
}
//运行结果&#xff1a;
//1 2 3 4 6 7 8 9

四、插入排序
算法思路&#xff1a;
1、从第一个元素开始&#xff0c;该元素可以认为已经被排序&#xff1b;
2、取出下一个元素&#xff0c;在已经排序的元素序列中从后向前扫描&#xff1b;
3、如果该元素&#xff08;已排序&#xff09;大于新元素&#xff0c;将该元素移到下一位置&#xff1b;
4、重复步骤3&#xff0c;直到找到已排序的元素小于或者等于新元素的位置&#xff1b;
5、将新元素插入到该位置后&#xff1b;
6、重复步骤2~5。
这里写图片描述
代码&#xff1a;

public class ChaRuPaiXu {
public static void main(String[] args) {
int array[] &#61; {1,2,4,3,9,7,8,6};
int index &#61; 0;
int current &#61; 0;
for (int i &#61; 1; i index &#61; i - 1; //左边的排是排好序的
current &#61; array[i]; //表示当前取到的扑克牌
while (index >&#61; 0 && array[index] > current) { //如果左边的排比取到的排大则右移
array[index &#43; 1] &#61; array[index];
index--;
}
array[index &#43; 1] &#61; current; //直到该手牌比抓到的牌小(或二者相等)&#xff0c;将抓到的牌插入到该手牌右边
}
for( int i &#61; 0 ; i System.out.print(array[i]&#43;" ");
}
}
}
//运行结果&#xff1a;
//1 2 3 4 6 7 8 9

五、希尔排序
算法思路&#xff1a;
1、选择一个增量序列t1&#xff0c;t2&#xff0c;…&#xff0c;tk&#xff0c;其中ti>tj&#xff0c;tk&#61;1&#xff1b;
2、按增量序列个数k&#xff0c;对序列进行k 趟排序&#xff1b;
3、每趟排序&#xff0c;根据对应的增量ti&#xff0c;将待排序列分割成若干长度为m 的子序列&#xff0c;分别对各子表进行直接插入排序。仅增量因子为1 时&#xff0c;整个序列作为一个表来处理&#xff0c;表长度即为整个序列的长度
这里写图片描述
常用的h序列由Knuth提出&#xff0c;该序列从1开始&#xff0c;通过如下公式产生&#xff1a;
h &#61; 3 * h &#43;1
反过来程序需要反向计算h序列&#xff0c;应该使用
h &#61; ( h - 1 ) / 3
代码&#xff1a;

public class XiErPaiXu {
public static void main(String[] args) {
int array[] &#61; {1,2,4,3,9,7,8,6};
int h &#61; 0;
int length &#61; array.length;
while( h <&#61; length ){ //计算首次步长
h &#61; 3 * h &#43; 1;
}
while( h >&#61; 1 ){
for( int i &#61; h;i int j &#61; i - h; //左边的一个元素
int get &#61; array[i]; //当前元素
while( j >&#61; 0 && array[j] > get ){ //左边的比当前大&#xff0c;则左边的往右边挪动
array[j&#43;h] &#61; array[j];
j &#61; j - h;
}
array[j &#43; h] &#61; get; //挪动完了之后把当前元素放进去
}
h &#61; ( h - 1 ) / 3;
}
for( int i &#61; 0 ; i System.out.print(array[i]&#43;" ");
}
}
}
//运行结果&#xff1a;
//1 2 3 4 6 7 8 9

六、归并排序
算法思路&#xff1a;
该算法是采用分治法&#xff08;Divide and Conquer&#xff09;的一个非常典型的应用。将已有序的子序列合并&#xff0c;得到完全有序的序列&#xff1b;即先使每个子序列有序&#xff0c;再使子序列段间有序。若将两个有序表合并成一个有序表&#xff0c;称为2-路归并。
1、把长度为n的输入序列分成两个长度为n/2的子序列&#xff1b;
2、对这两个子序列分别采用归并排序&#xff1b;
3、将两个排序好的子序列合并成一个最终的排序序列。
这里写图片描述
代码&#xff1a;

/*
k表示最终i和j比较之后最终需要放的位置
i和j用来表示当前需要考虑的元素
left表示最左边的元素
right表示最右边的元素
middle表示中间位置元素&#xff0c;放在第一个已经排好序的数组的最后一个位置
*/

public class GuiBingPaiXu {
/*******************测试************************/
public static void main(String[] args) {
int[] nums &#61; { 2, 7, 8, 3, 1, 6, 9, 0, 5, 4 , 9 , 19 ,12,16,14,12,22,33 };
mergeSort(nums , 0 , nums.length - 1 );
System.out.println(Arrays.toString(nums));
}
/********************算法************************/
/*
arr&#xff1a;要处理的数组
l&#xff1a;开始位置
r&#xff1a;结束位置
递归对arr[ l ... r ]范围的元素进行排序
*/

private static void mergeSort(int[] arr,int left,int right){
if( right - left <&#61; 10 ){ //当数据很少的时候使用插入排序算法
ChaRuPaiXu.ChaRuPaiXuFa2( arr , left ,right);
return;
}
int middle &#61; ( left &#43; right ) / 2; //计算中点位置
mergeSort( arr , left , middle ); //不断地对数组的左半边进行对边分
mergeSort( arr , middle&#43;1 , right ); //不断地对数组的右半边进行对半分
if( arr[middle] > arr[middle&#43;1] ) //当左边最大的元素都比右边最小的元素还小的时候就不用归并了
merge( arr , left , middle , right ); //最后将已经分好的数组进行归并
}
//将arr[ l... mid ]和arr[ mid ... r ]两部分进行归并
/*
|2, 7, 8, 3, 1 | 6, 9, 0, 5, 4|
*/

private static void merge(int[] arr, int left, int mid, int right) {
int arr1[] &#61; new int[ right - left &#43; 1 ]; //定义临时数组
for( int i &#61; left ; i <&#61; right ; i&#43;&#43; ) //将数组的元素全部复制到新建的临时数组中
arr1[ i - left ] &#61; arr[ i ];
int i &#61; left;
int j &#61; mid &#43; 1; //定义两个索引
for( int k &#61; left;k <&#61; right ; k&#43;&#43;){
if( i > mid ) //如果左边都比较完了
{
arr[ k ] &#61; arr1[ j - left ]; //直接将右边的元素都放进去
j&#43;&#43;;
}
else if( j > right ){ //右边都比较完了
arr[ k ] &#61; arr1 [i - left ]; //直接将左边的元素放进去
i&#43;&#43;;
}
else if( arr1[ i-left ] arr[ k ] &#61; arr1[ i - left];
i&#43;&#43;;
}
else
{
arr[ k ] &#61; arr1[ j - left];
j&#43;&#43;;
}
}
}
}

七、快速排序
算法思路&#xff1a;
通过一趟排序将待排记录分隔成独立的两部分&#xff0c;其中一部分记录的关键字均比另一部分的关键字小&#xff0c;则可分别对这两部分记录继续进行排序&#xff0c;以达到整个序列有序。
快速排序使用分治法来把一个串&#xff08;list&#xff09;分为两个子串&#xff08;sub-lists&#xff09;。
1、从数列中挑出一个元素&#xff0c;称为 “基准”&#xff08;pivot&#xff09;&#xff1b;
2、重新排序数列&#xff0c;所有元素比基准值小的摆放在基准前面&#xff0c;所有元素比基准值大的摆在基准的后面&#xff08;相同的数可以到任一边&#xff09;。在这个分区退出之后&#xff0c;该基准就处于数列的中间位置。这个称为分区&#xff08;partition&#xff09;操作&#xff1b;
3、递归地&#xff08;recursive&#xff09;把小于基准值元素的子数列和大于基准值元素的子数列排序。
这里写图片描述
代码&#xff1a;

public class KuaiSuPaiXu {
public static void main(String[] args){
int array[] &#61; {1,2,4,3,9,7,8,6};
quickSort(array,0,array.length-1);
for( int i &#61; 0 ; i System.out.print(array[i]&#43;" ");
}
}
private static void quickSort(int[] arr,int l,int r){
if( l >&#61; r ) return;
int p &#61; partition(arr,l,r); //找到中间位置
quickSort(arr,l,p-1);
quickSort(arr,p&#43;1,r);
}
private static int partition(int[] arr,int l,int r){
int v &#61; arr[l]; //取出第一个元素
int j &#61; l; //j表示小于第一个元素和大于第一个元素的分界点
for( int i &#61; l &#43; 1;i <&#61; r;i&#43;&#43; ){
//将所有小于第一个元素的值的元素全部都放到它的左边
if( arr[i] //如果当前元素小于v&#xff0c;则交换
swap(arr,i,j&#43;1);
j&#43;&#43;;
}
}
swap(arr,l,j); //将第一个元素和中间的元素进行交换
return j;
}
}
//运行结果&#xff1a;
//1 2 3 4 6 7 8 9

八、堆排序
算法思路&#xff1a;
堆排序&#xff08;Heapsort&#xff09;是指利用堆这种数据结构所设计的一种排序算法。堆积是一个近似完全二叉树的结构&#xff0c;并同时满足堆积的性质&#xff1a;即子结点的键值或索引总是小于&#xff08;或者大于&#xff09;它的父节点。
这里写图片描述
最大堆要求节点的元素都要不小于其孩子&#xff0c;最小堆要求节点元素都不大于其左右孩子
那么处于最大堆的根节点的元素一定是这个堆中的最大值.

public class DuiPaiXu {
public static void main(String[] args) {
int A[]&#61;{49,38,65,97,76,13,27,49,78,34,12,64,5,4,62,99,98,54,56,17,18,23,34,15,35,25,53,51};
HeapSort(A, A.length);
System.out.println(Arrays.toString(A));
}
public static void Swap(int A[], int i, int j)
{
int temp &#61; A[i];
A[i] &#61; A[j];
A[j] &#61; temp;
}
public static void Heapify(int A[], int i, int size) // 从A[i]向下进行堆调整
{
int left_child &#61; 2 * i &#43; 1; // 左孩子索引
int right_child &#61; 2 * i &#43; 2; // 右孩子索引
int max &#61; i; // 选出当前结点与其左右孩子三者之中的最大值
if (left_child A[max])
max &#61; left_child;
if (right_child A[max])
max &#61; right_child;
if (max !&#61; i)
{
Swap(A, i, max); // 把当前结点和它的最大(直接)子节点进行交换
Heapify(A, max, size); // 递归调用&#xff0c;继续从当前结点向下进行堆调整
}
}
public static int BuildHeap(int A[], int n) // 建堆&#xff0c;时间复杂度O(n)
{
int heap_size &#61; n;
for (int i &#61; heap_size / 2 - 1; i >&#61; 0; i--) // 从每一个非叶结点开始向下进行堆调整
Heapify(A, i, heap_size);
return heap_size;
}
public static void HeapSort(int A[], int n)
{
int heap_size &#61; BuildHeap(A, n); // 建立一个最大堆
while (heap_size > 1) // 堆&#xff08;无序区&#xff09;元素个数大于1&#xff0c;未完成排序
{
// 将堆顶元素与堆的最后一个元素互换&#xff0c;并从堆中去掉最后一个元素
// 此处交换操作很有可能把后面元素的稳定性打乱&#xff0c;所以堆排序是不稳定的排序算法
Swap(A, 0, --heap_size);
Heapify(A, 0, heap_size); // 从新的堆顶元素开始向下进行堆调整&#xff0c;时间复杂度O(logn)
}
}
}

Java面试的完整博客目录如下&#xff1a;Java笔试面试目录
这里写图片描述




推荐阅读
  • Netty框架中运用Protobuf实现高效通信协议
    在Netty框架中,通过引入Protobuf来实现高效的通信协议。为了使用Protobuf,需要先准备好环境,包括下载并安装Protobuf的代码生成器`protoc`以及相应的源码包。具体资源可从官方下载页面获取,确保版本兼容性以充分发挥其性能优势。此外,配置好开发环境后,可以通过定义`.proto`文件来自动生成Java类,从而简化数据序列化和反序列化的操作,提高通信效率。 ... [详细]
  • SQLite数据库CRUD操作实例分析与应用
    本文通过分析和实例演示了SQLite数据库中的CRUD(创建、读取、更新和删除)操作,详细介绍了如何在Java环境中使用Person实体类进行数据库操作。文章首先阐述了SQLite数据库的基本概念及其在移动应用开发中的重要性,然后通过具体的代码示例,逐步展示了如何实现对Person实体类的增删改查功能。此外,还讨论了常见错误及其解决方法,为开发者提供了实用的参考和指导。 ... [详细]
  • HBase Java API 进阶:过滤器详解与应用实例
    本文详细探讨了HBase 1.2.6版本中Java API的高级应用,重点介绍了过滤器的使用方法和实际案例。首先,文章对几种常见的HBase过滤器进行了概述,包括列前缀过滤器(ColumnPrefixFilter)和时间戳过滤器(TimestampsFilter)。此外,还详细讲解了分页过滤器(PageFilter)的实现原理及其在大数据查询中的应用场景。通过具体的代码示例,读者可以更好地理解和掌握这些过滤器的使用技巧,从而提高数据处理的效率和灵活性。 ... [详细]
  • 本文介绍了如何利用Apache POI库高效读取Excel文件中的数据。通过实际测试,除了分数被转换为小数存储外,其他数据均能正确读取。若在使用过程中发现任何问题,请及时留言反馈,以便我们进行更新和改进。 ... [详细]
  • 在托管C++中开发应用程序时,遇到了如何声明和操作字符串数组的问题。本文详细探讨了字符串数组在托管C++中的应用与实现方法,包括声明、初始化、遍历和常见操作技巧,为开发者提供了实用的参考和指导。 ... [详细]
  • Java SE 文件操作类详解与应用
    ### Java SE 文件操作类详解与应用#### 1. File 类##### 1.1 File 类概述File 类是 Java SE 中用于表示文件和目录路径名的对象。它提供了丰富的方法来操作文件和目录,包括创建、删除、重命名文件,以及获取文件属性和信息。通过 File 类,开发者可以轻松地进行文件系统操作,如检查文件是否存在、读取文件内容、列出目录下的文件等。此外,File 类还支持跨平台操作,确保在不同操作系统中的一致性。 ... [详细]
  • 开发笔记:深入解析Android自定义控件——Button的72种变形技巧
    开发笔记:深入解析Android自定义控件——Button的72种变形技巧 ... [详细]
  • Java中处理NullPointerException:getStackTrace()方法详解与实例代码 ... [详细]
  • 本文介绍了UUID(通用唯一标识符)的概念及其在JavaScript中生成Java兼容UUID的代码实现与优化技巧。UUID是一个128位的唯一标识符,广泛应用于分布式系统中以确保唯一性。文章详细探讨了如何利用JavaScript生成符合Java标准的UUID,并提供了多种优化方法,以提高生成效率和兼容性。 ... [详细]
  • 在Eclipse中批量转换Java源代码文件的编码格式从GBK到UTF-8是一项常见的需求。通过编写简单的Java代码,可以高效地实现这一任务。该方法不仅适用于Java文件,还可以用于其他类型的文本文件编码转换。具体实现可以通过导入`java.io.File`类来操作文件系统,从而完成批量转换。此外,建议在转换过程中添加异常处理机制,以确保代码的健壮性和可靠性。 ... [详细]
  • 探索偶数次幂二项式系数的求和方法及其数学意义 ... [详细]
  • 设计实战 | 10个Kotlin项目深度解析:首页模块开发详解
    设计实战 | 10个Kotlin项目深度解析:首页模块开发详解 ... [详细]
  • 本文深入探讨了CGLIB BeanCopier在Bean对象复制中的应用及其优化技巧。相较于Spring的BeanUtils和Apache的BeanUtils,CGLIB BeanCopier在性能上具有显著优势。通过详细分析其内部机制和使用场景,本文提供了多种优化方法,帮助开发者在实际项目中更高效地利用这一工具。此外,文章还讨论了CGLIB BeanCopier在复杂对象结构和大规模数据处理中的表现,为读者提供了实用的参考和建议。 ... [详细]
  • 基址获取与驱动开发:内核中提取ntoskrnl模块的基地址方法解析
    基址获取与驱动开发:内核中提取ntoskrnl模块的基地址方法解析 ... [详细]
  • 在处理大图片时,PHP 常常会遇到内存溢出的问题。为了避免这种情况,建议避免使用 `setImageBitmap`、`setImageResource` 或 `BitmapFactory.decodeResource` 等方法直接加载大图。这些函数在处理大图片时会消耗大量内存,导致应用崩溃。推荐采用分块处理、图像压缩和缓存机制等策略,以优化内存使用并提高处理效率。此外,可以考虑使用第三方库如 ImageMagick 或 GD 库来处理大图片,这些库提供了更高效的内存管理和图像处理功能。 ... [详细]
author-avatar
吴秋仪6_913
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有