排序 算法(第4版)

news2025/1/24 10:44:53

本博客参考算法(第4版):算法(第4版) - LeetBook - 力扣(LeetCode)全球极客挚爱的技术成长平台

本文用Java实现相关算法。

我们关注的主要对象是重新排列数组元素的算法,其中每个元素都有一个主键。排序算法的目标就是将所有元素的主键按照某种方式排列(通常是按照大小或是字母顺序)。排序后索引较大的主键大于等于索引较小的主键。元素和主键的具体性质在不同的应用中千差万别。在 Java 中,元素通常都是对象,对主键的抽象描述则是通过一种内置的机制,如Comparable接口。

大多数情况下,我们的排序代码只会通过两个方法操作数据:less() 方法对元素进行比较exch() 方法将元素交换位置。exch() 方法的实现很简单,通过 Comparable 接口实现 less() 方法也不困难。将数据操作限制在这两个方法中使得代码的可读性和可移植性更好,更容易验证代码的正确性、分析性能以及排序算法之间的比较。

在本文主键全是整数,因此不做接口实现,直接实现相关函数。

private static void exch(int[] arr, int i, int j) {
    int temp = arr[i]; arr[i] = arr[j]; arr[j] = temp;
}
private static boolean less(int pre, int suf) {
    return pre < suf;
}

排序默认从小到大升序

初级排序算法

选择排序

一种最简单的排序算法是这样的:首先,找到数组中最小的那个元素,其次,将它和数组的第一个元素交换位置(如果第一个元素就是最小元素那么它就和自己交换)。再次,在剩下的元素中找到最小的元素,将它与数组的第二个元素交换位置。如此往复,直到将整个数组排序。这种方法叫做选择排序,因为它在不断地选择剩余元素之中的最小者。

代码实现:

public class lab {
    private static void exch(int[] arr, int i, int j) {
        int temp = arr[i]; arr[i] = arr[j]; arr[j] = temp;
    }
    private static boolean less(int pre, int suf) {
        return pre < suf;
    }
    public static void sort(int[] arr) {
        int N = arr.length; // 数组长度
        for(int i = 0; i < N; ++i) {
            int pos = i; // 最小元素下标
            for(int j = i + 1; j < N; ++j) {
                // 如果有比当前最小元素a[j]小的,则更新最小元素下标
                if(less(arr[j], arr[pos])) pos = j;
            }
            // 交换当前位置i 和最小元素下标
            exch(arr, i, pos);
        }
    }
    public static int[] a = {5, 4, 1, 56, 3};
    // 测试
    public static void main(String[] args) {
        sort(a);
        for(int i : a) System.out.println(i);
    }
}

交换元素的代码写在内循环之外,每次交换都能排定一个元素,因此交换的总次数是 N。所以算法的时间效率取决于比较的次数。

对于长度为 N 的数组,选择排序需要大约 N^2/2 次比较和N次交换。

选择排序两个鲜明的特点:

  • 运行时间和输入无关
  • 数据移动是最少的

插入排序

通常人们整理桥牌的方法是一张一张的来,将每一张牌插入到其他已经有序的牌中的适当位置。在计算机的实现中,为了给要插入的元素腾出空间,我们需要将其余所有元素在插入之前都向右移动一位。这种算法叫做插入排序。

和选择排序不同的是,插入排序所需的时间取决于输入中元素的初始顺序。例如,对一个很大且其中的元素已经有序(或接近有序)的数组进行排序将会比对随机顺序的数组或是逆序数组进行排序要快得多。

代码实现:

public class lab {
    private static void exch(int[] arr, int i, int j) {
        int temp = arr[i]; arr[i] = arr[j]; arr[j] = temp;
    }
    private static boolean less(int pre, int suf) {
        return pre < suf;
    }
    public static void sort(int[] arr) {
        int N = arr.length; // 数组长度
        for(int i = 1; i < N; ++i) {
            // 将 a[i] 插入到 a[i-1]、a[i-2]、a[i-3]...之中
            for(int j = i; j > 0 && less(a[j], a[j - 1]); --j) {
                exch(arr, j, j - 1);
            }
        }
    }
    public static int[] a = {5, 4, 1, 56, 3};
    // 测试
    public static void main(String[] args) {
        sort(a);
        for(int i : a) System.out.println(i);
    }
}

对于随机排序的长度N且主键不重复的数组,平均情况下插入需要N * N / 4次比较及N * N / 4次交换。最坏是N * N / 2次比较和交换。最好情况是N - 1次比较和0次交换(已有序数组)。

插入排序对部分有序的数组很有效果,例如:

  • 数组中每个元素距离它的最终位置都不远;
  • 一个有序的大数组接一个小数组;
  • 数组中只有几个元素的位置不正确。

插入排序需要的交换操作和数组中倒置的数量相同,需要的比较次数大于等于倒置的数量,小于等于倒置的数量加上数组的大小再减一。

希尔排序

优化插入排序。希尔排序为了加快速度简单地改进了插入排序,交换不相邻的元素以对数组的局部进行排序,并最终用插入排序将局部有序的数组排序。

希尔排序的思想是使数组中任意间隔为 h 的元素都是有序的。这样的数组被称为 h 有序数组。对于任意以 1 结尾的 h 序列,我们都能够将数组排序,这就是希尔排序。

{80%}

实现希尔排序的一种方法是对于每个 h,用插入排序将 h 个子数组独立地排序。但因为子数组是相互独立的,一个更简单的方法是在 h- 子数组中将每个元素交换到比它大的元素之前去(将比它大的元素向右移动一格)。只需要在插入排序的代码中将移动元素的距离由 1 改为 h 即可。这样,希尔排序的实现就转化为了一个类似于插入排序但使用不同增量的过程。

public class lab {
    private static void exch(int[] arr, int i, int j) {
        int temp = arr[i]; arr[i] = arr[j]; arr[j] = temp;
    }
    private static boolean less(int pre, int suf) {
        return pre < suf;
    }
    public static void sort(int[] arr) {
        int N = arr.length; // 数组长度
        int h = 1;
        while(h < N / 3) h = 3 * h + 1; //  // 1, 4, 13, 40, 121, 364, 1093, ... 这样效果更优
        while(h >= 1) {
            // 虽然是两个for循环,但是加在一起是O(N)
            for(int i = h; i < N; ++i) {
                // 插入思想
                for(int j = i; j >= h && less(a[j], a[j - h]); j -= h) {
                    exch(arr, j, j - h);
                }
            }
            h /= 3;
        }
    }
    public static int[] a = {5, 4, 1, 56, 3};
    // 测试
    public static void main(String[] args) {
        sort(a);
        for(int i : a) System.out.println(i);
    }
}

归并排序

归并排序,可以先(递归地)将它分成两半分别排序,然后将结果归并起来。归并排序有原地归并、自顶向下、自顶向上这几种方法,本文只讲自顶向下的方法。

public class lab {
    private static void exch(int[] arr, int i, int j) {
        int temp = arr[i]; arr[i] = arr[j]; arr[j] = temp;
    }
    private static boolean less(int pre, int suf) {
        return pre < suf;
    }
    public static void sort(int[] arr, int lo, int hi) {
        if(lo >= hi) return ;
        int mid = (lo + hi) / 2;
        // 将数组分为两部分,并对这两部分进行排序
        sort(arr, lo, mid); sort(arr, mid + 1, hi);
        merge(arr, lo, mid, hi);
    }
    // 将两个有序数组进行合并
    public static void merge(int[] arr, int lo, int mid, int hi) {
        int i = lo, j = mid + 1, k = 0;
        while(i <= mid && j <= hi) {
            if(less(arr[i], arr[j])) tmp[k++] = arr[i++];
            else tmp[k++] = arr[j++];
        }
        while(i <= mid) tmp[k++] = arr[i++];
        while(j <= hi) tmp[k++] = arr[j++];
        for(i = lo, k = 0; i <= hi; ++i, ++k) arr[i] = tmp[k];
    }
    public static int[] a = {5, 4, 1, 56, 3};
    public static int[] tmp = new int[5]; // 辅助数组
    // 测试
    public static void main(String[] args) {
        sort(a, 0, 4);
        for(int i : a) System.out.println(i);
    }
}

该算法需要N * lg(N) / 2 N * lg(N)次比较。最多访问数组6 * N * lg(N)次。

快速排序

快速排序可能是应用最广泛的排序算法了。快速排序流行的原因是它实现简单、适用于各种不同的输入数据且在一般应用中比其他排序算法都要快得多。快速排序引人注目的特点包括它是原地排序(只需要一个很小的辅助栈)。

快速排序是一种分治的排序算法。它将一个数组分成两个子数组,将两部分独立地排序。**快速排序和归并排序是互补的:归并排序将数组分成两个子数组分别排序,并将有序的子数组归并以将整个数组排序;而快速排序将数组排序的方式则是当两个子数组都有序时整个数组也就自然有序了。**归并排序的递归调用发生在处理整个数组之前;快速排序是递归调用发生在处理整个数组之和。

public class lab {
    private static void exch(int[] arr, int i, int j) {
        int temp = arr[i]; arr[i] = arr[j]; arr[j] = temp;
    }
    private static boolean less(int pre, int suf) {
        return pre < suf;
    }
    public static void sort(int[] arr, int lo, int hi) {
        if(lo >= hi) return ;
        // 进行切分,下标为partidx的元素在整个数组中是已经排序好的
        int partidx = partition(arr, lo, hi);
        // 对左右两部分进行排序
        sort(arr, lo, partidx - 1); sort(arr, partidx + 1, hi);
    }
    // 得到第j位是有序的对于整个数组来说
    public static int partition(int[] arr, int lo, int hi) {
        int i = lo, j = hi + 1; // 左右指针
        int v = arr[lo]; // 切分元素的值
        while(true) {
            // 扫描左右,检查扫描是否结束并交换元素
            while(less(arr[++i], v)) if(i == hi) break;
            while(less(v, arr[--j])) if(j == lo) break;
            if(i >= j) break;
            exch(arr, i, j);
        }
        exch(arr, lo, j);
        // 将这个数组第j位最终态找到
        return j;
    }
    public static int[] a = {5, 4, 1, 56, 3};
    public static int[] tmp = new int[5]; // 辅助数组
    // 测试
    public static void main(String[] args) {
        sort(a, 0, 4);
        for(int i : a) System.out.println(i);
    }
}

优先队列

优先队列是一种数据结构:支持快速的删除最大元素和插入元素。

优先队列可以用二叉堆实现。二叉堆,本质上是一种完全二叉树。

分类:二叉堆分为最大堆和最小堆两种类型,最大堆和最小堆分别又可称为大顶堆和小顶堆。最大堆中,任何一个父节点的值都大于或等于它的左、右孩子节点的值;最小堆中,任何一个父节点的值都小于或等于它的左、右孩子节点的值。

{%}

用数组(堆)实现的完全二叉树的结构是很严格的,但它的灵活性已经足以让我们高效地实现优先队列。用它们我们将能实现对数级别的插入元素和删除最大元素的操作。利用在数组中无需指针即可沿树上下移动的便利和以下性质,算法保证了对数复杂度的性能。

在有序化的过程中我们会遇到两种情况。当某个结点的优先级上升(或是在堆底加入一个新的元素)时,我们需要由下至上恢复堆的顺序。当某个结点的优先级下降(例如,将根结点替换为一个较小的元素)时,我们需要由上至下恢复堆的顺序。首先,我们会学习如何实现这两种辅助操作,然后再用它们实现插入元素和删除最大元素的操作。

由下向上的堆有序(上浮)

如果堆的有序状态因为某个结点变得比它的父结点更大而被打破,那么我们就需要通过交换它和它的父结点来修复堆。交换后,这个结点比它的两个子结点都大(一个是曾经的父结点,另一个比它更小,因为它是曾经父结点的子结点),但这个结点仍然可能比它现在的父结点更大。我们可以一遍遍地用同样的办法恢复秩序,将这个结点不断向上移动直到我们遇到了一个更大的父结点。

private void swim(int k)
{
   while (k > 1 && less(k/2, k))
   {
      exch(k/2, k);
      k /= 2;
   }
}

由上到下的堆有序(下沉)

private void sink(int k) {
    while(2 * k <= N) {
        int j = 2 * k;
        if(j < N && less(j, j + 1)) ++j;
        if(!less(k, j)) break;
        exch(k, j);
        k = j;
    }
}

堆排序

public class lab {
    private static void exch(int i, int j) {
        int temp = a[i]; a[i] = a[j]; a[j] = temp;
    }
    private static boolean less(int i, int j) {
        return a[i] < a[j];
    }
    public static void sort() {
        int N = a.length - 1;
        for(int k = N / 2; k >= 1; --k) {
            sink(k, N);
        }
        while(N > 1) {
            exch(1, N--);
            sink(1, N);
        }
    }
    private static void swim(int k) {
        while(k > 1 && less(a[k / 2], a[k])) {
            exch(k / 2, k);
            k /= 2;
        }
    }
    private static void sink(int k, int N) {
        while(2 * k <= N) {
            int j = 2 * k;
            if(j < N && less(j, j + 1)) ++j;
            if(!less(k, j)) break;
            exch(k, j);
            k = j;
        }
    }
    public static int[] a = {0, 5, 4, 1, 56, 3};
    // 测试
    public static void main(String[] args) {
        sort();
        for(int i : a) System.out.println(i);
    }
}

排序算法的时间复杂度和稳定性

在这里插入图片描述

算法(第4版) - LeetBook - 力扣(LeetCode)全球极客挚爱的技术成长平台

二叉堆的节点插入、删除以及构建过程_二叉堆插入-CSDN博客

常用十大排序算法-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1205759.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

NSSCTF题库——web

[SWPUCTF 2021 新生赛]gift_F12 f12后ctrlf找到flag [SWPUCTF 2021 新生赛]jicao——json_decode() 加密后的格式 $json {"a":"php","b":"mysql","c":3}; json必须双引号传输 构造&#xff1a;GET里json{"x"…

【JUC】三、集合的线程安全

文章目录 1、ArrayList集合线程安全问题分析2、解决方式一&#xff1a;Vector或synchronizedList( )3、解决方式二&#xff1a;CopyOnWriteArrayList 写时复制4、HashSet集合线程不安全的分析与解决5、HashMap集合线程不安全的分析与解决 1、ArrayList集合线程安全问题分析 对…

TLP超线程技术

在实现IPL指令级并行的同时实现TLP(Thread Level Parallelism)线程级并行实现多线程有两种主要的方法超线程即同时多线程&#xff0c;在单个处理器或单个核中设置了两套线程状态部件&#xff0c;共享高速缓存和功能部件当两个线程同时需要某个资源时&#xff0c;其中一个线程必…

VuePress介绍及使用指南

VuePress是一个基于Vue.js的静态网站生成工具&#xff0c;它专注于以Markdown为中心的项目文档。VuePress具有简单易用的特性&#xff0c;同时提供了强大的自定义和扩展性。在本文中&#xff0c;我们将介绍VuePress的基本概念&#xff0c;并提供一个简单的使用指南。 什么是Vue…

【C语言】

C语言 1. C语言基础1.1 数据类型和占位符1.2 异或1.3 关键字1.4 const1.5 extern1.6 typedef1.7 static1.8 左值和右值1.9 位进行操作赋值 2. C指针3. 二维数组和指针4. 函数传递二维数组4.1 形参给出第二维的长度。4.2 形参声明为指向数组的指针。4.3 形参声明为指针的指针。 …

快速掌握队列的基础知识

目录 队列的特点基于链表实现队列用栈实现队列用队列实现栈 队列是一种线性数据结构&#xff0c;它只允许在一边进行插入操作&#xff08;队尾&#xff09;&#xff0c;另一边进行删除操作&#xff08;队头&#xff09;。插入操作称为入队&#xff0c;删除操作称为出队。队列遵…

【网络奇缘】我和英特网再续前缘

&#x1f308;个人主页: Aileen_0v0&#x1f525;系列专栏: 一见倾心,再见倾城 --- 计算机网络~&#x1f4ab;个人格言:"没有罗马,那就自己创造罗马~" 目录 计算机网络的概念 计算机网络的功能 ⭐1.数据通信 ⭐2.资源共享 ⭐3.分布式处理 ⭐4.提高可靠性 ⭐…

Java17新增特性

前言 前面的文章&#xff0c;我们对Java9、Java10、Java11、Java12 、Java13、Java14、Java15、Java16 的特性进行了介绍&#xff0c;对应的文章如下 Java9新增特性 Java10新增特性 Java11新增特性 Java12新增特性 Java13新增特性 Java14新增特性 Java15新增特性 Java16新增特…

Eclipse打包Springboot项目

首先&#xff0c;在pom.xml文件中添加配置&#xff0c;修改mainClass主函数&#xff1a; <build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId><configurat…

【电路笔记】-戴维南定理(Thevenin‘s Theorem)

戴维南定理&#xff08;Thevenin’s Theorem&#xff09; 文章目录 戴维南定理&#xff08;Thevenins Theorem&#xff09;1、概述与定义2、戴维南模型确定3、一些线性电路的戴维南模型3.1 单电压源3.2 单电流源3.3 多电流/电压源 4、结论 在本文中&#xff0c;我们将介绍一种强…

LLaMA模型之中文词表的蜕变

在目前的开源模型中&#xff0c;LLaMA模型无疑是一颗闪亮的⭐️&#xff0c;但是相对于ChatGLM、BaiChuan等国产大模型&#xff0c;其对于中文的支持能力不是很理想。原版LLaMA模型的词表大小是32K&#xff0c;中文所占token是几百个左右&#xff0c;这将会导致中文的编解码效率…

Python---综合案例:通讯录管理系统---涉及点:列表、字典、死循环

需求&#xff1a; 开个一个通讯录的管理系统&#xff0c;主要用于实现存储班级中同学的信息&#xff08;姓名、年龄、电话&#xff09; 涉及点&#xff1a;列表、字典、死循环 相关链接&#xff1a;Python--列表及其应用场景---增、删、改、查。-CSDN博客 Python---字典---…

ip数据包

数据报文格式 首部 版本&#xff08;Version&#xff09; 版本字段占4bit&#xff0c;通信双方使用的版本必须一致。对于IPv4&#xff0c;字段的值是4。 首部长度&#xff08;Internet Header Length&#xff0c; IHL&#xff09; 占4bit&#xff0c;首部长度说明首部有多少…

ubutun上编译出现undefined reference to symbol ‘dladdr@@GLIBC_2.2.5‘的错误

作者&#xff1a;朱金灿 来源&#xff1a;clever101的专栏 为什么大多数人学不会人工智能编程&#xff1f;>>> ubutun上编译一段C程序&#xff0c;出现错误&#xff1a; /usr/bin/ld: /tmp/ccghh3FJ.o: undefined reference to symbol ‘dladdrGLIBC_2.2.5’ //lib/…

前端---CSS的盒模型

文章目录 什么是盒模型&#xff1f;设置边框设置内边距设置外边距块级元素水平居中 什么是盒模型&#xff1f; 页面上的每个HTML元素都是一个一个的“盒子”&#xff0c;这些盒子由&#xff1a;内容、内边距、边框、外边距组成。 我们可以和住的房子联系起来&#xff0c;更好…

HarmonyOS开发(二):TypeScript入门

1、编程语言介绍 ArkTS是HarmonyOS主推的应用开发语言&#xff0c;它是在TypeScript语言的基础之上&#xff0c;匹配ArkUI框架&#xff0c;扩展了声明式UI、状态管理等相应的能力&#xff0c;让开发者以更简洁、更自然的方式开发跨端应用。 ArkTS、TypeScript和JavaScript之间…

打开 Chrome 的 「内存节省程序」开关和关闭硬件加速

不知道从什么时候开始&#xff0c;应该是最近1个月&#xff0c;感觉 Mac 浏览器总是占用很高的 CPU&#xff0c;多开一些标签页&#xff0c;或者浏览器窗口&#xff0c;相互切换时系统就会变得无响应&#xff0c;从 Chrome 浏览器里找到一个「内存节省程序」的配置&#xff0c;…

数据校验:Spring Validation

Spring Validation概述 在开发中&#xff0c;我们经常遇到参数校验的需求&#xff0c;比如用户注册的时候&#xff0c;要校验用户名不能为空、用户名长度不超过20个字符、手机号是合法的手机号格式等等。如果使用普通方式&#xff0c;我们会把校验的代码和真正的业务处理逻辑耦…

二维码智慧门牌管理系统升级解决方案:运营可视化之道

文章目录 前言一、系统概述二、数据可视化与运营决策 前言 随着科技的飞速发展和人们生活水平的提高&#xff0c;传统的门牌管理系统已经无法满足现代社会的需求。在这个信息化、智能化的时代&#xff0c;一款升级版的二维码智慧门牌管理系统应运而生&#xff0c;它将以全新的…

手机能做静态二维码吗?用手机做二维码的教程

现在手机上有很多的功能&#xff0c;能够帮助我们应对日常生活中的各种问题&#xff0c;那么如果我们想要在手机上生成一个静态二维码&#xff0c;大家知道该怎么来操作吗&#xff1f;一般制作二维码需要专业的二维码生成工具才可以完成制作&#xff0c;那么下面小编来给大家分…