冒泡排序、插入排序、快速排序、堆排序、希爾排序、歸并排序

目錄

        • 冒泡排序
        • 插入排序
        • 快速排序(未優化版本)
        • 快速排序(優化版本)
        • 堆排序
        • 希爾排序
        • 歸并排序
        • 各排序時間消耗對比

冒泡排序

冒泡排序核心邏輯就是對數組從第一個位置開始進行遍歷,如果發現該元素比下一個元素大,則交換位置,如果不大,就拿著下一個比該元素大的元素向下繼續比較,以此類推一直比較到最后,每一輪下來就會把當前最大的元素放到最后所以每一輪會確定一個最大元素。

    public void bubbleSort(int[] array) {for (int i = 0; i < array.length; i++) {int flag = 0;for (int j = 1; j < array.length-i; j++) {if (array[j - 1] > array[j]) {swap(array,j-1,j);flag = 1;}}if (flag == 0) {break;}}}

這里做了一個優化,當某一輪發現一個元素都沒有移動的時候,就表示這個數組已經是有序的了,不理解的可以舉一個已經有序的例子推敲一下,所以定義一個flag如果為0就表示一整倫下來都沒有發生過移動直接跳出循環,排序完成。

時間復雜度:O(N^2)

穩定性:穩定

空間復雜度:O(1)

插入排序
    public void insertSort(int[] array) {for (int i = 1; i < array.length; i++) {int j = i - 1;int tmp = array[i];for (; j >= 0; j--) {if (array[j] >= tmp) {array[j+1] = array[j];} else {break;}}array[j+1] = tmp;}}

時間復雜度:O(N^2)

穩定性:穩定

空間復雜度:O(1)

快速排序(未優化版本)
	public void quickSort (int[] array) {quick(array,0,array.length-1);}private void quick(int[] array, int start, int end) {if (start >= end) {return;}// 一個是霍爾法一個是挖坑法,兩個都可int part = partition1(array,start,end); // 霍爾法//int part = partition2(array,start,end); // 挖坑法quick(array,start,part-1);quick(array,part+1,end);}private int partition1(int[] array, int left, int right) {int tmp = array[left];int tmpIndex = left;while (left < right) {while (left < right && array[right] >= tmp) {right--;}while (left < right && array[left] <= tmp) {left++;}swap(array,left,right);}swap(array,tmpIndex,left);return left;}private int partition2(int[] array, int left ,int right) {int tmp = array[left];while (left < right) {while (left < right && array[right] >= tmp) {right--;}array[left] = array[right];while (left < right && array[left] <= tmp) {left++;}array[right] = array[left];}array[left] = tmp;return left;}

最好時間復雜度:O(N*logN)

最壞時間復雜度:O(N^2)

空間復雜度:O(logN)

穩定性:不穩定

快速排序(優化版本)
	public void quickSort (int[] array) {quick(array,0,array.length-1);}private void quick(int[] array, int start, int end) {if (start >= end) {return;}// 優化1(如果是少量的數據,直接插入排序速度是很快的,所以我們用直接插入排序處理數量小于等于15個的情況,不要小看這個優化,因為quickSort是樹形的遞歸模式,在最后一層會有大量的數據存在,這個時候我們直接不進行遞歸直接進行插入排序可以節省大量的開銷)if (end - start + 1 <= 15) {insertWithQuick(array,start,end);return;}// 優化2(三數取中優化,如果當一個數組比較趨于有序的情況下,那么構造出來的二叉樹就是變成了一個鏈表,所以才會有時間復雜度區域O(N^2),我們只要打破這個順序就好,我們直接去一個較為中間的值作為基準值就能均分了)int midIndex = findMidIndex(array,start,end);swap(array,start,midIndex);// 一個是霍爾法一個是挖坑法,兩個都可int part = partition1(array,start,end); // 霍爾法//int part = partition2(array,start,end); // 挖坑法quick(array,start,part-1);quick(array,part+1,end);}private int partition1(int[] array, int left, int right) {int tmp = array[left];int tmpIndex = left;while (left < right) {while (left < right && array[right] >= tmp) {right--;}while (left < right && array[left] <= tmp) {left++;}swap(array,left,right);}swap(array,tmpIndex,left);return left;}private int partition2(int[] array, int left ,int right) {int tmp = array[left];while (left < right) {while (left < right && array[right] >= tmp) {right--;}array[left] = array[right];while (left < right && array[left] <= tmp) {left++;}array[right] = array[left];}array[left] = tmp;return left;}// 使用插入排序進行優化private void insertWithQuick(int[] array, int start, int end) {for (int i = start; i <= end; i++) {int j = i - 1;int tmp = array[i];for (; j >= start; j--) {if (array[j] >= tmp) {array[j+1] = array[j];} else {break;}}array[j+1] = tmp;}}// 使用三數取中找到中值下標private int findMidIndex(int[] array, int start, int end) {int midIndex = start + (end - start) / 2;if (array[start] > array[end]) {// end  startif (array[midIndex] < array[end]) {return end;} else if (array[midIndex] > array[start]) {return start;} else {return midIndex;}} else {// start  endif (array[midIndex] < array[start]) {return start;} else if (array[midIndex] > array[end]) {return end;} else {return midIndex;}}}

優化后的時間復雜度將會無限趨近于O(N*logN),空間復雜度不變

堆排序
    public void heapSort (int[] array) {createBigHeap(array);int end = array.length-1;while (end >= 0) {swap(array,0,end);end--;siftDown(array,0,end);}}private void createBigHeap(int[] array) {for (int parent = (array.length-2); parent >= 0; parent--) {siftDown(array,parent,array.length-1);}}private void siftDown(int[] array, int parent, int end) {int child = parent * 2 + 1;while (child <= end) {if (child + 1 <= end && array[child+1] > array[child]) {child++;}if (array[child] > array[parent]) {swap(array,child,parent);parent = child;child = child * 2 + 1;} else {break;}}}

時間復雜度:O(N*logN)

空間復雜度:O(logN)

穩定性:不穩定

希爾排序
    public void shallSort (int[] array) {int gap = array.length / 2;while (gap > 0) {shall(array,gap);gap /= 2;}}private void shall(int[] array, int gap) {for (int i = gap; i < array.length; i++) {int tmp = array[i];int j = i - gap;for (; j >= 0; j -= gap) {if (array[j] >= tmp) {array[j + gap] = array[j];} else {break;}}array[j+gap] = tmp;}}

希爾排序就是直接插入排序的升級版本

時間復雜度:不好計算業界公認的大概是O(N^1.3)

空間復雜度:O(1)

穩定性:不穩定

歸并排序
    public void mergeSort(int[] array) {mergeFun(array,0,array.length-1);}private void mergeFun(int[] array, int start, int end) {if (start >= end) {return;}int midIndex = start + (end - start) / 2;mergeFun(array,start,midIndex);mergeFun(array,midIndex+1,end);// mergemerge(array, start, end, midIndex);}private void merge(int[] array, int start, int end, int midIndex) {int rs = midIndex + 1 , k = 0, tmpIndex = start;int[] tmpArray = new int[end - start +1];while (start <= midIndex && rs <= end) {if (array[start] < array[rs]) {tmpArray[k++] = array[start++];} else {tmpArray[k++] = array[rs++];}}while (start <= midIndex) {tmpArray[k++] = array[start++];}while (rs <= end) {tmpArray[k++] = array[rs++];}for (int j = tmpIndex; j <= end; j++) {array[j] = tmpArray[j - tmpIndex];}}

時間復雜度:O(N*logN)

空間復雜度:O(N)

穩定性:穩定

各排序時間消耗對比

為了更直觀的觀察每個排序有時間效率,我將這幾個排序進行了一個對比,先隨機生成要給數組大小為100000的隨機數,隨機數范圍是0-10000,之后分別計算各個排序所消耗的時間單位是毫秒

public static void main(String[] args) {Sort sort = new Sort();// 原數組int[] array = createRandomNumber(100_000,10_000);// copy數組int[] array1 = Arrays.copyOf(array, array.length);long start1 = System.currentTimeMillis();sort.quickSort(array1);System.out.println("quickSort : " + (System.currentTimeMillis()-start1) );int[] array2 = Arrays.copyOf(array, array.length);long start2 = System.currentTimeMillis();sort.shallSort(array2);System.out.println("shallSort : " + (System.currentTimeMillis()-start2) );int[] array3 = Arrays.copyOf(array, array.length);long start3 = System.currentTimeMillis();sort.heapSort(array3);System.out.println("heapSort : " + (System.currentTimeMillis()-start3) );int[] array4 = Arrays.copyOf(array, array.length);long start4 = System.currentTimeMillis();sort.mergeSort(array4);System.out.println("mergeSort : " + (System.currentTimeMillis()-start4) );int[] array5 = Arrays.copyOf(array, array.length);long start5 = System.currentTimeMillis();sort.insertSort(array5);System.out.println("insertSort : " + (System.currentTimeMillis()-start5) );int[] array6 = Arrays.copyOf(array, array.length);long start6 = System.currentTimeMillis();sort.bubbleSort(array6);System.out.println("bubbleSort : " + (System.currentTimeMillis()-start6) );}private static int[] createRandomNumber (int size, int range) {int[] array = new int[size];for (int i = 0; i < size; i++) {array[i] = new Random().nextInt(range);}return array;}

在這里插入圖片描述

所以大家還是少用冒泡吧,雖然簡單但是效率感人~ 快排還是牛掰

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/pingmian/77007.shtml
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/77007.shtml
英文地址,請注明出處:http://en.pswp.cn/pingmian/77007.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

JavaScript:表單及正則表達式驗證

今天我要介紹的是在JavaScript中關于表單驗證內容的知識點介紹&#xff1a; 關于表單驗證&#xff0c;我接下來則直接將內容以及效果顯示出來并作注解&#xff0c;這樣可以清晰看見這個表達驗證的妙用&#xff1a; <form id"ff" action"https://www.baidu.…

天元證券|調倉曝光!首批科技基金一季報出爐

4月15日&#xff0c;中歐基金、永贏基金、長城基金等公募基金公司旗下部分權益類基金產品一季報出爐。 券商中國記者梳理發現&#xff0c;永贏信息產業智選混合主要聚焦信息技術領域布局&#xff0c;前十大重倉股中9只股票屬于信息技術行業&#xff0c;合計占基金資產凈值比例達…

SpringAI版本更新:向量數據庫不可用的解決方案!

Spring AI 前兩天&#xff08;4.10 日&#xff09;更新了 1.0.0-M7 版本后&#xff0c;原來的 SimpleVectorStore 內存級別的向量數據庫就不能用了&#xff0c;Spring AI 將其全部源碼刪除了。 此時我們就需要一種成本更低的解決方案來解決這個問題&#xff0c;如何解決呢&…

Sklearn入門之datasets的基本用法

、 Sklearn全稱:Scipy-toolkit Learn是 一個基于scipy實現的的開源機器學習庫。它提供了大量的算法和工具&#xff0c;用于數據挖掘和數據分析&#xff0c;包括分類、回歸、聚類等多種任務。本文我將帶你了解并入門Sklearn下的datasets在機器學習中的基本用法。 獲取方式 pi…

優化 Dockerfile 性能之實踐(Practice of Optimizing Dockerfile Performance)

優化 Dockerfile 性能之實踐 構建 Docker 鏡像時&#xff0c;Dockerfile 的性能會顯著影響構建過程的效率。經過優化的 Dockerfile 可以縮短構建時間、最小化鏡像大小并提高整體容器性能。在本文中&#xff0c;我們將探討優化 Dockerfile 性能的最佳實踐。 盡量減少層數 影響…

出現 ERR_CERT_COMMON_NAME_INVALID | 301 302 重定向的解決方法

目錄 前言1. 問題所示2. 原理分析3. 解決方法前言 ?? 找工作,來萬碼優才:?? #小程序://萬碼優才/r6rqmzDaXpYkJZF 爬蟲神器,無代碼爬取,就來:bright.cn 1. 問題所示 執行代碼時,出現如下提示: GET https://xxxx/admin-api/system

C語言 —— 指尖躍遷 刻印永恒 - 文件操作

目錄 1. 什么是文件 1.1 程序文件 1.2 數據文件 1.3 文件名 2. 二進制文件和文本文件 3. 文件的打開與關閉 3.1 流和標準流 3.2 文件指針 3.3 文件的打開與關閉 fopen fclose 4. 文件的順序讀寫 4.1 fgetc和fputc fgetc fputc 4.2 fgets和fputs fgets fputs…

用css給div列表加個序號

用 CSS 的 counter 相關屬性來為列表添加序號。以下是具體的代碼&#xff0c;我將以 HTML 文件的形式提供&#xff0c;并且會運行展示效果&#xff1a; .as-div {// counter-reset: my-counter; /* 計數器名稱是my-counter */// counter-reset: small-apple; /* 計數器名稱是s…

Rust : 關于*const () 與type erase

*const () 可以替代泛型&#xff0c;更加靈活。 一、 代碼 //use std::mem::transmute; trait Work {fn process(&self); } struct Foo(String);impl Work for Foo {fn process(&self) {println!("process work from Foo : {}", self.0);} } struct Bar(S…

【專題刷題】雙指針(二)

&#x1f4dd;前言說明&#xff1a; 本專欄主要記錄本人的基礎算法學習以及LeetCode刷題記錄&#xff0c;按專題劃分每題主要記錄&#xff1a;&#xff08;1&#xff09;本人解法 本人屎山代碼&#xff1b;&#xff08;2&#xff09;優質解法 優質代碼&#xff1b;&#xff…

吉爾吉斯斯坦工商會代表團赴齊河德瑞新能源汽車考察

德州齊河&#xff0c;2025年4月15日電 時中美貿易突變之際&#xff0c;乘國家一帶一路之風。 展中國新能源之宏圖&#xff0c;塑國貿體系之新方向。 今日上午&#xff0c;吉爾吉斯斯坦共和國工商會代表團一行三人受邀抵達濟南&#xff0c;開啟對德瑞新能源科技有限公司&…

【記錄condapack打包環境到超算上順利運行】

以安裝CLRNet為例子 本地Linux系統上的操作步驟。 由于官方的安裝包的步驟&#xff0c;執行condapack的時候會報錯&#xff0c;所以使用以下步驟進行安裝包。 安裝其他 Python 依賴包 pip install -r requirements.txt? 二、構建并打包項目&#xff08;核心步驟&#xff…

Windows OpenUtau-v0.1.529-開源歌曲合成軟件[提供MIDI編輯、歌詞調整、音色修改 等功能,音樂創作者的必備工具]

Windows OpenUtau 鏈接&#xff1a;https://pan.xunlei.com/s/VONy_Refvo6_813Ig--nu5_rA1?pwdejzc# 引擎&#xff08;Resampler&#xff09;和拼接器&#xff08;Wavtool&#xff09;是UTAU協議中音頻處理的兩大組件。前端編輯器通過調用引擎和拼接器&#xff0c;對音頻進行…

虛擬卡可以解決訂閱 ChatGPT 時無法付款的問題

在全球掀起 AI 熱潮的今天&#xff0c;因為工作的需要有些朋友要用ChatGPT&#xff0c;它也成為了不少人日常學習、工作、創作和編程的得力助手。然而&#xff0c;不少用戶在嘗試訂閱 ChatGPT Plus&#xff08;付費版&#xff09;時&#xff0c;卻遇到了一個令人頭疼的問題——…

設計模式之狀態模式:優雅管理對象行為變化

引言 狀態模式&#xff08;State Pattern&#xff09;是一種行為型設計模式&#xff0c;它允許對象在其內部狀態改變時改變它的行為&#xff0c;使對象看起來似乎修改了它的類。狀態模式將狀態轉移邏輯和狀態相關行為封裝在獨立的狀態類中&#xff0c;完美解決了復雜條件判斷問…

【算法】歸并排序

算法系列七&#xff1a;歸并排序 一、歸并排序的遞歸探尋 1.思路 2.搭建 2.1設計過掉不符情況&#xff08;在最底層時&#xff09; 2.2查驗能實現基礎排序&#xff08;在最底層往上點時&#xff09; 2.3跳轉結果繼續往上回搭 3.實質 4.實現 二、遞歸的調用棧 1.遞歸的…

線束線纜從二維設計到虛擬驗證全流程解決方案

一、傳統設計中的痛點 線纜的開發設計是橫跨多專業多學科的龐大工程&#xff0c;通常會劃分為幾大階段逐次推進&#xff0c;由于每個階段的工作任務不同&#xff0c;所以在不同設計階段使用的工具也完全不同&#xff0c;由此導致整個設計流程中工程師常常要跨平臺協作&#xf…

【智駕中的大模型 -1】自動駕駛場景中的大模型

1. 前言 我們知道&#xff0c;大模型現在很火爆&#xff0c;尤其是 deepseek 風靡全球后&#xff0c;大模型毫無疑問成為為中國新質生產力的代表。百度創始人李彥宏也說&#xff1a;“2025 年可能會成為 AI 智能體爆發的元年”。 隨著科技的飛速發展&#xff0c;大模型的影響…

個人博客系統后端 - 注冊登錄功能實現指南

一、功能概述 個人博客系統的注冊登錄功能包括&#xff1a; 用戶注冊&#xff1a;新用戶可以通過提供用戶名、密碼、郵箱等信息創建賬號用戶登錄&#xff1a;已注冊用戶可以通過用戶名和密碼進行身份驗證&#xff0c;獲取JWT令牌身份驗證&#xff1a;使用JWT令牌訪問需要認證…

投行交易與風控系統的消費側冪等架構設計與實戰

1.背景和痛點 1.1 資金操作敏感性場景 核心需求&#xff1a; 交易唯一性&#xff1a;資金類操作必須保證全局唯一執行計算原子性&#xff1a;風控指標計算需具備事務性特征審計追溯&#xff1a;所有操作需保留完整冪等軌跡 1.2 業務損失統計 二、技術挑戰與架構設計 2.1 分…