Java的时间复杂度和空间复杂度和常见排序

news2024/9/22 15:40:20

        

目录

一丶时间复杂度

二丶空间复杂度

三丶Java常见排序

       1. 冒泡排序(Bubble Sort)

        2.插入排序(Insertion Sort)

         3.希尔排序(Shell Sort)

         4.选择排序(Selection Sort)

          5.堆排序(Heap Sort)

           6.归并排序(Merge Sort)

           7.快速排序(Quick Sort)

        8.计数排序(Counting Sort)、桶排序(Bucket Sort) 和 基数排序(Radix Sort)


        简介:时间复杂度和空间复杂度是评估算法性能的两个重要指标,他们分别用于衡量算法执行时间长短和算法所存储空间大小;

一丶时间复杂度

        时间复杂度:他描述了算法执行所需时间和数据规模之间的关系。具体来说,时间复杂度是算法中基本操作语句执行的次数,这次次数随着数据规模的增大而增大。时间复杂度通常用大O表示法(Big O notation)来表示,他忽略了常数因子和低阶项,只保留最高阶项,从而简洁明了的表示出算法的时间增长趋势;例如

  • O(1):表示算法的执行时间是固定,与输入规模无关;
  • O(log n):表示算法的执行时间与输入规模的对数成正比;
  • O(n):表示线性时间复杂度,算法的执行时间与输入规模成线性比例增长;
  • O(n log n):表示算法的执行时间与输入规模的线性对数比例增长;
  • O(n^2):表示平方时间复杂度,算法的执行时间与输入规模的平方成比例增长。

通过分析算法的时间复杂度,可以评估算法的性能,优化算法的效率,从而提高程度的执行速度。

二丶空间复杂度

        空间复杂度:它描述了算法在运行过程中临时占用存储空间的大小与数据规模之间的关系。空间复杂度也是用大O表示法来表示,他衡量的是算法运行过程中额外消耗的存储空间。例如

  • O(1):表示算法的空间复杂度是固定的,与输入规模无关;
  • O(log n):表示线性空间复杂度,算法所需内存与输入规模成线性比例增长
  • O(n^2):表示平方空间复杂度,算法所需内存与输入规模的平方成比例增长。

通过分析算法的空间复杂度,可以避免内存的浪费,提高空间利用率,从而降低算法执行成本,提高程序性能;

三丶Java常见排序

       1. 冒泡排序(Bubble Sort)

        原理:通过重复地遍历要排序的数列,一次比较两个元素,如果他们的顺序错误就把他们交换过来。遍历数列的工作是重复的进行直到没有再需要交换,也就是说该数列已经排序完成。

        特点:简单,稳定,单效率低。时间复杂度O(n^2),空间复杂度O(1);

    //冒泡排序
    public static void bubbleSort(int[] arr) {
        int n = arr.length;
        for (int i = 0; i < n-1; i++) {
            for (int j = 0; j < n-i-1; j++) {
                if (arr[j] > arr[j+1]) {
                    int temp = arr[j];
                    arr[j] = arr[j+1];
                    arr[j+1] = temp;
                }
            }
        }
    }
        2.插入排序(Insertion Sort)

        原理:通过构建有序序列,对于未排序数据,在已排序序序列中从后向前扫描,找到相应位置插入;

        特点:稳定排序,适用于少量数据的排序,但是数据接近有序时效率较高。时间复杂度最好的情况下O(n),最坏的情况下O(n^2);空间复杂度O(1);

    //直接插入排序
    public static void insertionSort(int[] arr) {
        int n = arr.length;
        for (int i = 1; i < n; i++) {
            //取出第二个数据,默认已经排序
            int key = arr[i];
            //获取前以为数据的索引
            int j = i-1;
        /* 将 arr[0..i-1] 中大于 key 的元素移动到其当前位置的前 1 个位置*/
            while (j >=0 && arr[j] > key) {
                arr[j+1] = arr[j];
                j = j-1;
            }
            arr[j+1] = key;
        }
    }
         3.希尔排序(Shell Sort)

        原理:是插入排序的一种更高效的改进版本。希尔排序又称为缩小量排序,它将整个待排序的记录序列分割成为若干子序列分别进行直接插入排序,待整个序列中的记录基本有序,在对全体记录进行直接插入排序;

        特点:不稳定的排序,时间复杂度依赖于增量序列的选择,但平均性能优于直接插入排序;

时间复杂度:O(n^1.3),空间复杂度O(1);

    //直接排序
    public static void shellSort(int[] arr) {
        int n = arr.length;
        for (int gap = n/2; gap > 0; gap /= 2) {
            for (int i = gap; i < n; i += 1) {
                int temp = arr[i];
                int j;
                for (j = i; j >= gap && arr[j - gap] > temp; j -= gap) {
                    arr[j] = arr[j - gap];
                }
                arr[j] = temp;
            }
        }
    }
         4.选择排序(Selection Sort)

        原理:首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾;

        特点:不稳定排序,时间复杂度:O(n^2),空间复杂度:O(1);

    //选择排序
    public void selectionSort(int[] arr) {
        int n = arr.length;
        for (int i = 0; i < n-1; i++) {
            int min_idx = i;
            for (int j = i+1; j < n; j++) {
                if (arr[j] < arr[min_idx]) {
                    min_idx = j;
                }
            }
            // 将找到的 Minimum 元素与第一个元素交换
            int temp = arr[min_idx];
            arr[min_idx] = arr[i];
            arr[i] = temp;
        }
    }
          5.堆排序(Heap Sort)

        原理:是指利用堆这种数据结构所设计的一种排序算法。堆积是一个近似完全二叉树的结构,并同时满足堆积的性质:即子节点的键值或索引总是小于(或大于)他的父节点;

        特点:不稳定排序,时间复杂度:O(n log n),空间复杂度:O(1);

// 构建最大堆(辅助函数)  
void buildMaxHeap(int arr[]) {  
    int n = arr.length;  
    for (int i = n / 2 - 1; i >= 0; i--)  
        heapify(arr, n, i);  
}  
  
// 调整给定的堆  
void heapify(int arr[], int n, int i) {  
    int largest = i; // 初始化最大为根  
    int l = 2 * i + 1; // 左 = 2*i + 1  
    int r = 2 * i + 2; // 右 = 2*i + 2  
  
    // 如果左子节点大于根  
    if (l < n && arr[l] > arr[largest])  
        largest = l;  
  
    // 如果右子节点是最大值  
    if (r < n && arr[r] > arr[largest])  
        largest = r;  
  
    // 如果最大值不是根  
    if (largest != i) {  
        int swap = arr[i];  
        arr[i] = arr[largest];  
        arr[largest] = swap;  
  
        // 递归地堆化受影响的子树  
        heapify(arr, n, largest);  
    }  
}  
  
// 堆排序函数  
void heapSort(int arr[]) {  
    int n = arr.length;  
    buildMaxHeap(arr);  
  
    // 一个个从堆顶取出元素  
    for (int i = n - 1; i > 0; i--) {  
        // 移动当前根到末尾  
        int temp = arr[0];  
        arr[0] = arr[i];  
        arr[i] = temp;  
  
        // 调用max heapify on the reduced heap  
        heapify(arr, i, 0);  
    }  
}
           6.归并排序(Merge Sort)

        原理:是建立在归并操作上的一种有效的排序算法。该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序;

        特点:稳定排序,时间复杂度:O(n log n),空间复杂度:O(n);

public void mergeSort(int[] arr, int l, int r) {  
    if (l < r) {  
        // Same as (l+r)/2, but avoids overflow for  
        // large l and h  
        int m = l+(r-l)/2;  
  
        // Sort first and second halves  
        mergeSort(arr, l, m);  
        mergeSort(arr, m+1, r);  
  
        merge(arr, l, m, r);  
    }  
}  
  
// 合并两个已排序的数组部分  
void merge(int arr[], int l, int m, int r) {  
    // Find sizes of two subarrays to be merged  
    int n1 = m - l + 1;  
    int n2 = r - m;  
  
    /* Create temp arrays */  
    int L[] = new int[n1];  
    int R[] = new int[n2];  
  
    /*Copy data to temp arrays*/  
    for (int i=0; i<n1; ++i)  
        L[i] = arr[l + i];  
    for (int j=0; j<n2; ++j)  
        R[j] = arr[m + 1+ j];  
  
    /* Merge the temp arrays */  
  
    // Initial indexes of first and second subarrays  
    int i = 0, j = 0;  
  
    // Initial index of merged subarray array  
    int k = l;  
    while (i < n1 && j < n2) {  
        if (L[i] <= R[j]) {  
            arr[k] = L[i];  
            i++;  
        } else {  
            arr[k] = R[j];  
            j++;  
        }  
        k++;  
    }  
  
    /* Copy remaining elements of L[] if any */  
    while (i < n1) {  
        arr[k] = L[i];  
        i++;  
        k++;  
    }  
  
    /* Copy remaining elements of R[] if any */  
    while (j < n2) {  
        arr[k] = R[j];  
        j++;  
        k++;  
    }  
}
           7.快速排序(Quick Sort)

        原理:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列;

        特点:平均时间复杂度:O(n log n),但是最坏的情况下时间复杂度会变成O(n^2)。空间复杂度取决于递归深度,平均为O(n log n),但最坏情况下需要O(n)的额外空间;

public void quickSort(int[] arr, int low, int high) {  
    if (low < high) {  
        // pi is partitioning index, arr[p] is now  
        // at right place  
        int pi = partition(arr, low, high);  
  
        // Separately sort elements before  
        // partition and after partition  
        quickSort(arr, low, pi - 1);  
        quickSort(arr, pi + 1, high);  
    }  
}  
  
// 该方法用于分区数组,返回分区索引  
int partition(int arr[], int low, int high) {  
    int pivot = arr[high];  
    int i = (low - 1); // index of smaller element  
    for (int j = low; j < high; j++) {  
        // If current element is smaller than or  
        // equal to pivot  
        if (arr[j] <= pivot) {  
            i++;  
  
            // swap arr[i] and arr[j]  
            int temp = arr[i];  
            arr[i] = arr[j];  
            arr[j] = temp;  
        }  
    }  
  
    // swap arr[i+1] and arr[high] (or pivot)  
    int temp = arr[i + 1];  
    arr[i + 1] = arr[high];  
    arr[high] = temp;  
  
    return i + 1;  
}
        8.计数排序(Counting Sort)桶排序(Bucket Sort) 和 基数排序(Radix Sort)
  • 这些排序算法是非比较型排序算法,其排序效率在某些情况下会高于比较型排序算法。它们各自适用于一定范围的数据,如计数排序适用于一定范围内的整数排序,桶排序和基数排序则适用于外部排序和大数据排序。

 

结尾:喜欢的朋友点个赞吧!!! 

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

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

相关文章

视频汇聚平台LntonAIServer视频质量诊断功能--偏色检测与噪声检测

随着视频监控技术的不断进步&#xff0c;视频质量成为了决定监控系统性能的关键因素之一。LntonAIServer新增的视频质量诊断功能&#xff0c;特别是偏色检测和噪声检测&#xff0c;进一步强化了视频监控系统的可靠性和实用性。下面我们将详细介绍这两项功能的技术细节、应用场景…

【Leetcode】1-5

1 两数之和 1. 两数之和 - 力扣&#xff08;LeetCode&#xff09; 和为目标值 target 就是在找 target - nums[i] 利用 哈希表 查找只需要 O(1) class Solution {public int[] twoSum(int[] nums, int target) {HashMap<Integer, Integer> hm new HashMap<>();…

科研绘图系列:R语言柱状图分布(histogram plot)

介绍 柱状图(Bar Chart)是一种常用的数据可视化图表,用于展示和比较不同类别或组的数据。它通过在二维平面上绘制一系列垂直或水平的柱子来表示数据的大小,每个柱子的长度或高度代表一个数据点的数值。柱状图非常适合于展示分类数据的分布和比较。柱状图的特点: 直观比较…

使用ChatGPT半小时撰写优质学术报告,详细使用指南

大家好,感谢关注。我是七哥,一个在高校里不务正业,折腾学术科研AI实操的学术人。关于使用ChatGPT等AI学术科研的相关问题可以和作者七哥(yida985)交流,多多交流,相互成就,共同进步,为大家带来最酷最有效的智能AI学术科研写作攻略。经过数月爆肝,终于完成学术AI使用教…

C++ | Leetcode C++题解之第386题字典序排数

题目&#xff1a; 题解&#xff1a; class Solution { public:vector<int> lexicalOrder(int n) {vector<int> ret(n);int number 1;for (int i 0; i < n; i) {ret[i] number;if (number * 10 < n) {number * 10;} else {while (number % 10 9 || numbe…

打包部署之---》Xshell使用不了如何部署

前端打包发布有很多种&#xff0c;常用 xshell 连接服务器打包发布&#xff0c;但是小编发现最近小编电脑上的Xshell7出现了一个问题&#xff0c;一直报50003错误&#xff0c;说是不是最新版本&#xff1b;让你升级 可是点击确定以后确提示小编已经是最新版本; 这个时候小编选择…

【Java】面向对象基础(创建类,认识构造器,this关键字)

文章目录 前言一、创建类二、面向对象的基础&#xff08;认识构造器&#xff09;三、this关键字总结 前言 学习Java面向对象的基础。 一、创建类 1、在创建class文件的时候&#xff0c;文件夹名称跟第一个创建出来的类名是一样的。 二、面向对象的基础&#xff08;认识构造器…

数据结构---双向链表(内存泄露相关知识)

一、内存泄露 内存泄露&#xff08;Memory Leak&#xff09;是指程序中已动态分配的堆内存由于某种原因程序未释放或无法释放&#xff0c;造成系统内存的浪费&#xff0c;导致程序运行速度减慢甚至系统崩溃等严重后果。内存泄漏是程序设计中常见的错误之一&#xff0c;其特点包…

C语言学习笔记 Day16(C10文件管理--下)

Day16 内容梳理&#xff1a; C语言学习笔记 Day14&#xff08;文件管理--上&#xff09;-CSDN博客 C语言学习笔记 Day15&#xff08;文件管理--中&#xff09;-CSDN博客 目录 Chapter 10 文件操作 10.5 文件状态 10.6 文件的随机读写 fseek()、rewind() &#xff08;1&…

chapter13-常用类——(String类)——day15

目录 466-String结构剖析 467-String结构剖析 468-String测试题1 469-String测试题2 470-String对象特性1 471-String对象特性2 472-String常用方法1 473-String常用方法2 474-String常用方法3 466-String结构剖析 1、接口Serializabel&#xff0c;说明String对象可以串…

智汇云舟受邀参加2024第四届国产水科学数值模型开发创新与技术应用研讨会,并成为“科技智水产业联盟”创始成员

在数字化浪潮的推动下&#xff0c;智慧水利作为国家战略的重要组成部分&#xff0c;正迎来前所未有的发展机遇。8月27-29日&#xff0c;由浙江贵仁信息科技股份有限公司主办、浙江省水利学会协办的“2024第四届国产水科学数值模型开发创新与技术应用研讨会”在杭州白马湖建国饭…

【深度学习】向量化

1. 什么是向量化 向量化通常是消除代码中显示for循环语句的技巧&#xff0c;在深度学习实际应用中&#xff0c;可能会遇到大量的训练数据&#xff0c;因为深度学习算法往往在这种情况下表现更好&#xff0c;所以代码的运行速度非常重要&#xff0c;否则如果它运行在一个大的数据…

英伟达AI超级计算机SuperPod:H100→GH200→GB200

英伟达的 DGX SuperPOD 是一台完整的数据中心级 AI 超级计算机&#xff0c;采用模块化的设计&#xff0c;支持不同规模大小的设计。每台超级计算机都在出厂前完成了搭建、布线和测试&#xff0c;从而大大加快了在用户数据中心的部署速度 。 NVIDIA DGX SuperPOD是下一代数据中心…

【设计模式】单例、工厂、策略、责任链模式

1.单例模式 单例模式确保某个类只有一个实例,主要使用有两种:懒汉式单例、饿汉式单例 单例模式有以下特点:   1、单例类只能有一个实例。   2、单例类必须自己创建自己的唯一实例。   3、单例类必须给所有其他对象提供这一实例。 使用场景: 数据库连接池:单例模式…

Python 算法交易实验87 QTV200日常推进-沪深300的交易量统计

说明 上一篇 讲到&#xff0c;就全市场的交易量来看&#xff0c;近3年也看不出很奇怪的地方&#xff1b;因此&#xff0c;交易量[支持度]不足而导致的策略失灵似乎也说不通。 为了进一步确认&#xff0c;我打算&#xff1a; 1 获取沪深300成分股&#xff0c;只统计成分股的交…

RDD、DataFrame、DataSet(Spark)

RDD、DataFrame、DataSet RDD (弹性分布式数据集)&#xff1a; 版本: Spark的初始版本&#xff0c;1.0开始提供。特性: RDD是Spark的基础数据结构&#xff0c;表示一个不可变的分布式对象集合。可以通过转换操作&#xff08;如map、filter、flatMap&#xff09;和行动操作&…

outlook设置规则后,规则无法自动运行的解决方案

outlook设置规则后&#xff0c;规则无法自动运行的解决方案 问题描述解决方案说明 问题描述 在outlook设置规则对邮件进行文件夹移动时&#xff0c;有时会出现规则无法自动运行的情况。出现这种情况有很多原因和解决方案&#xff0c;本文列出其中一种。 解决方案 outlook版本…

吴恩达深度学习笔记:卷积神经网络(Foundations of Convolutional Neural Networks)1.9-1.10

目录 第四门课 卷积神经网络&#xff08;Convolutional Neural Networks&#xff09;第一周 卷积神经网络&#xff08;Foundations of Convolutional Neural Networks&#xff09;1.9 池化层&#xff08;Pooling layers&#xff09;1.10 卷 积 神 经 网 络 示 例 &#xff08; …

JDBC的介绍

一 JDBC的简介 1.1 ODBC的出现 早期的数据库应用程序开发&#xff0c;因为没有通用的针对与数据库的编程接口&#xff0c;所以&#xff0c;开发人员需要学习相关数据库的API&#xff0c;才可以进行应用程序&#xff0c;这样增加了学习成本和开发周期。因此整个开发市场一直在呼…

QT creator堆栈窗体demo

创建项目选择QDialog,不要选QMainWindow #ifndef DIALOGSTACK_H #define DIALOGSTACK_H#include <QDialog> #include <QListWidget>//列表框 #include <QStackedWidget>//堆栈窗体 #include <QLabel>QT_BEGIN_NAMESPACE namespace Ui { class DialogSt…