”插入排序“”選擇排序“

文章目錄

  • 插入排序
    • 1. 直接插入排序(O(n^2))
      • 舉例1:
      • 舉例2:
      • 直插排序的"代碼"
      • 直插排序的“時間復雜度”
    • 2. 希爾排序(O(n^1.3))
        • 方法一
        • 方法二(時間復雜度更優)
  • 選擇排序
    • 堆排序
    • 直接選擇排序

我們學過冒泡排序,堆排序等等。(回顧一下:排升序,建大堆;排降序,建小堆。)

排序:所謂排序,就是使?串記錄,按照其中的某個或某些關鍵字的??,遞增或遞減的排列起來的
操作。

在生活中也會遇到很多排序:購物篩選排序(按照價格、綜合、銷量、距離等排序)、百度熱搜(根據熱搜程度)、院校排名等等。

在這里插入圖片描述

插入排序

1. 直接插入排序(O(n^2))

基本思想:將(待排序的記錄)按照(關鍵碼值的大小)逐個插入到(已經排好序的有序隊列)中。直到所有的記錄插入完為止,得到一個新的有序序列。

先將arr[end]指向:(有序隊列的最后一個數據),然后將(需要和有序數據進行比較的數據:即arr[end+1])暫時存儲在tmp中,接下來將arr[end]和tmp里的數據進行比較。
在這里插入圖片描述
注意在一次循環中,tmp的值是不變的(除非end++,才會使tmp的值改變(tmp=arr[end]))

舉例1:

【假設想要排升序】
在這里插入圖片描述

  • 現在arr[end]的值大于tmp,應該排在后面,所以現在將end的值放在end+1的位置,這個時候會發現end+1位置的值被覆蓋了,所以我們需要提前將它的值存儲在tmp中(這就是將值存儲在tmp中的原因)。接下來需要將tmp的值放在end的位置(這個是arr[end]的前面沒有其他值的情況,我們可以直接將tmp的值賦給arr[end])

在交換值之后,將end++,tmp的值為arr[end+1] (它不用修改,因為end已經修改了)
在這里插入圖片描述

  • 如果arr[end]<tmp,并不需要交換呢?(比如5和9)那就直接end++,緊接著tmp的值也變啦。

  • 緊接著又是將9和6交換,按照之前的方法(將arr[end]的值賦給arr[end+1],然后將tmp的值賦給arr[end]
    在這里插入圖片描述

  • 為什么我們還需要將交換之后的arr[end]的值和這個序列前面的值比較呢?------防止它比前面已經排好序的值小

接下來按照前面的方法,再將9和2進行比較

舉例2:

在這里插入圖片描述

直插排序的"代碼"

我們需要一個循環來控制end++,往后走;
還需要一個循環來控制:當把arr[end]賦值給arr[end+1]之后,需要將end- -,將前面的數值和tmp繼續比較。

在這里插入代碼片`void InsertSort(int* arr, int n)
{for (int i = 0; i < n - 1; i++)  //為什么是i<n-1?{int end = i;int tmp = arr[end + 1];while (end >= 0){if (arr[end] > tmp){arr[end + 1] = arr[end];end--;}else{break;}}arr[end + 1] = tmp;}
}`

在這里插入圖片描述

  • 為什么是i<n-1呢?
    因為tmp存儲的是arr[end+1],我們要確保它是最后一個不會越界,即end+1<n,又因為end=i,所以i+1<n,所以i<n-1

直插排序的“時間復雜度”

直接插入排序的時間復雜度是O(n^2)

不過O(n^2)是最差的情況。
在這里插入圖片描述

最差:想將降序----->升序 O(n^2)
最好:升序------->升序 O(n)

2. 希爾排序(O(n^1.3))

希爾排序就是對直接插入排序的一種優化。想要改善降序---->升序的時間復雜度。

希爾排序分為兩步:1.預排序 2.直接插入排序

我們可以將很長的一段數組序列變為n段較短的序列,然后對每段“單獨”進行直接插入排序(這個叫做預排序)。預排結束之后,小的數據大部分都在前半部分了(只是不按順序),接著我們再對整體的數組(已經趨于有序的數組)序列進行一次直插排序,這樣不僅完成了排序,還優化了時間復雜度。

方法一
  • 首先我們需要決定一個值:gap,然后將大段分為n個小段(每段的首和尾相隔gap個元素,即第一個值往后數gap個,那個數據是這段的最后一個元素)。

什么情況下預排:當gap>1的時候
什么情況下整段直插:當gap==1的時候

在這里插入圖片描述

  1. 從上圖可以知曉:要將一大段數據分為很多組數據,在尋找第二組數據時,需要將上次的首元素下標++。這一步由外層循環控制【遍歷每個子序列的起始位置(從 0 到 gap-1)】for(i=0;i<gap;i++)(為什么i<gap?)

  2. 接下來是每組數據中,知道了首元素,然后找這組數據中的剩余數據,即end+gap,這個由第二層循環控制:for(int j=0;j<n-gap;j+=gap);(為什么內層的j<n-gap?)

  3. 再里面的元素之間比較就沒有什么特別了,使用的是直接插入排序(它也有循環)。

【gap不能太大(若太大的話,會導致每組的數據比較少,但組數比較多),也不能太小(那樣的話每組數據會有很多,那這樣的話時間復雜度又變高了)】我們可以讓gap由元素個數決定,gap=n/3+1,為什么最后還要+1呢?(如果遇到2/3的情況,那商就是0,也就是gap=0,那間距為0,誰和誰比呢?)而我們最后需要讓gap=1,讓已經預排之后的整段數據進行直插排序,所以需要+1.

【gap等于多少,這段數據最終就會被分為幾組數據】如果n特別特別大,n/3之后也特大,先將+1給忽略掉

  • 之前我們說到為什么外層循環中i<gap而不是i<n呢?
    在這里插入圖片描述
    我們第一組數據的第二個元素是arr[end+gap]也就是arr[0+gap],既然arr[gap]在第一組數據中已經進行比較過一次,那之后就沒必要再將它比較一次了。大家觀察上圖會發現每個數據都進行比較過,如果外層用 i < n,會導致重復處理同一子序列,效率降低。

  • 還有一個問題是:為什么第二層循環需要j<n-gap,按照方法一個組中,每個數據它們都相鄰gap個(n=8,gap=3第一組元素下標:0,3,6,下標為6的元素后面還有1個元素,但是剩余元素個數不夠gap個,那就只能在這次的循環中將后面的元素舍棄。所以在找每組的數據時(即在第二層循環),循環結束的條件是最后一個元素后面的元素個數小于gap個(有剩余的元素或者剩余個數為0(也就是剛好后面沒有元素了)),也就是它的下標小于n-gap

在這里插入圖片描述

第一組的數據:7,4,1
第二組的數據:6,3
第三組的數據:5,2

  • 然后,先排第一組的數據,再排2,3組的數據

(1)初始情況:arr[end]指向第一個元素,tmp指向(與第一個相隔gap的元素).即tmp=arr[end+gap]

  • arr[end]>tmp,那么將arr[end]的值賦給tmp所指向的那個值,即arr[end+gap]=arr[end],賦值之后,將end=end-gap
  • arr[end]>tmp,不用修改值。直接將end+=gap
    對于每組數據的值比較的注釋
    (2)排完第二組的數據之后,外層的i++,到第二個數據【下標為1】(它作為第二組的首元素),然后進入第二層循環,在進行比較。

到此為止,這個進行了一趟比較

void ShellSort(int* arr, int n)
{int gap = n / 3 + 1;for (int i = 0; i < gap; i++){int j = i;for (j =i; j < n - gap; j += gap){int end = j;int tmp = arr[end + gap];while (end>=0)  //將這組數據的每個數據比較,while結束的條件:end>=0;{if (arr[end] > tmp){arr[end + gap] = arr[end];//想繼續和前面有序的數據比較,將end-gapend -= gap;}//到這里,arr[end]<tmp,那就不用賦值,直接將下標+gap,tmp的值隨之改變,繼續比較else {break;}}//在循環中,我們只是將大的數據往后賦值,那當初那個比較小的tmp賦給誰呢?//這組數據如何能比較結束(break)呢?肯定是此時的arr[end]并沒有tmp大,不需要交換了//所以,我們把tmp值賦值給之前一個比較的值(由于是end-=gap,所以上一個是end+gap)arr[end + gap] = tmp;}}
}

在這里插入圖片描述

在這里插入圖片描述

之后我們將gap不斷縮小,繼續比較,直至最后gap==1時,進行最后的,對整段進行直插
在這里插入圖片描述

void ShellSort(int* arr, int n)
{int gap = n;while (gap>1){gap = gap / 3 + 1;for (int i = 0; i < gap; i++){int j = i;for (j = i; j < n - gap; j += gap){int end = j;int tmp = arr[end + gap];while (end >= 0)  //將這組數據的每個數據比較,while結束的條件:end>=0;{printf("%d ", gap);if (arr[end] > tmp){arr[end + gap] = arr[end];//想繼續和前面有序的數據比較,將end-gapend -= gap;}//到這里,arr[end]<tmp,那就不用賦值,直接將下標+gap,tmp的值隨之改變,繼續比較else {break;}}//在循環中,我們只是將大的數據往后賦值,那當初那個比較小的tmp賦給誰呢?//這組數據如何能比較結束(break)呢?肯定是此時的arr[end]并沒有tmp大,不需要交換了//所以,我們把tmp值賦值給之前一個比較的值(由于是end-=gap,所以上一個是end+gap)arr[end + gap] = tmp;}}}
}

注意:為什么gap>1還可以對整段進行直插排序呢?
大家先想一想gap==1的條件:肯定是上一個gap>1因而進入了循環,進行了gap=gap/3+1,而gap/3=0,gap/3+1=1,接下來進行了直插。不會有其他意外。

n=10:
第一次,gap=n=10,大于1,進入循環之后,gap=gap/3+1=4,之后進行排序。
第二次,gap=4,大于1,進入循環,gap=gap/3+1=4/3+1=2,之后進行排序
注意gap=2的時候它仍然大于1 ,進入循環了,然后才進行的gap=gap/3+1,得到了gap=1,進行最后的直插

方法二(時間復雜度更優)

在上面的基礎上進行優化:
之前是將一組一組比,現在仍然是相隔gap個的兩個元素進行比較,但是當它倆比較完之后,我們不用找arr[0+gap]再隔gap的值。我們直接找arr[0]的下一個:arr[1],再找arr[1+gap],將他倆比較。那到什么時候停止呢?當i<n-gap時。
在這里插入圖片描述

void ShellSort(int* arr, int n)
{int gap = n;while (gap>1){gap = gap / 3 + 1;for (int i=0; i < n - gap; i++){int end = i;int tmp = arr[end + gap];while (end >= 0)  //將這組數據的每個數據比較,while結束的條件:end>=0;{if (arr[end] > tmp){arr[end + gap] = arr[end];//想繼續和前面有序的數據比較,將end-gapend -= gap;}//到這里,arr[end]<tmp,那就不用賦值,直接將下標+gap,tmp的值隨之改變,繼續比較else {break;}}//在循環中,我們只是將大的數據往后賦值,那當初那個比較小的tmp賦給誰呢?//這組數據如何能比較結束(break)呢?肯定是此時的arr[end]并沒有tmp大,不需要交換了//所以,我們把tmp值賦值給之前一個比較的值(由于是end-=gap,所以上一個是end+gap)arr[end + gap] = tmp;}}
}

在這里插入圖片描述
希爾排序的時間復雜度是:O(n^1.3)
在這里插入圖片描述

選擇排序

堆排序

這是堆排序的文章,點擊鏈接即可跳轉

直接選擇排序

這個名字中的“選擇”已經暴露了它的方法,直接選擇排序是從整個數組中(arr[0]到arr[n-1])選出來min或者max的元素,然后放在整個數組的第一個位置。然后再從arr[1]到arr[n-1]選出來min或max,放在第二個位置。【將min/max放在第一個位置,那原本的數據怎么辦,所以不是直接將這個值放在第一個位置,而是將兩個值交換】

  • 思路:

    1. 先設定兩個變量,begin和mini(它倆是下標),begin作為min或者max放置的位置的下標。mini作為遍歷數組時此時位置的下標。[需要知道跳出循環的條件,即放的位置<n]
      在這里插入圖片描述
  • 但是又一個bug是:每找一次最大/最小值,就需要將數組遍歷一遍(和冒泡排序一樣費時間O(n^2)),時間復雜度很大,我們進行優化。

在遍歷一遍數組的時候,我們可以將min和max同時找出來,將它們放在數組的兩端。這樣就可以省略一半的時間。【但需要注意的是,這個上面那個的循環結束條件就不一樣了,我們不用再遍歷后半部分,就是已經放好特定值的那部分】

void SelectSort(int* arr, int n)
{int begin=0;for (begin = 0; begin < n/2; begin++) //為什么begin<n/2呢?因為begin是前半部分的下標,只用占整個數組的一半{int maxi = begin;int mini = begin;//確定了這趟mini要放的位置,接下來這個循環用來遍歷找最小值/最大值,用j來遍歷int j = begin;for (j = begin+1; j < n-begin; j++)  //為什么j<n-begin//第一遍遍歷的時候,需要遍歷所有的//第二次遍歷,就不用遍歷最后一個,已經放了特定的值//第三次遍歷,倒數兩個都不用遍歷了{if (arr[j] < arr[mini])mini=j;if (arr[j] > arr[maxi])maxi=j;}//這一步是什么作用?if (maxi == begin){maxi = mini;}Swap(&arr[begin], &arr[mini]);Swap(&arr[n - 1 - begin], &arr[maxi]);}
}

其中這一步的作用是什么呢?

if (maxi == begin)
{maxi = mini;
}

舉個例子:
在這里插入圖片描述

這個的問題就在于:maxi剛好和begin重合了,而arr[begin]先一步和arr[mini]交換了,所以我們需要在交換之前處理一下:如果它倆重合了,那就先讓maxi走到mini的位置。
在這里插入圖片描述

if(maxi==begin)maxi=mini;

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

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

相關文章

FPGA_BD Block Design學習(一)

PS端開發流程詳細步驟 1.第一步&#xff1a;打開Vivado軟件&#xff0c;創建或打開一個工程。 2.第二步&#xff1a;在Block Design中添加arm核心&#xff0c;并將其配置為IP核。 3.第三步&#xff1a;配置arm核心的外設信息&#xff0c;如DDR接口、時鐘頻率、UART接口等。 …

【Python] pip制作離線包

制作離線安裝包是一種非常實用的方法&#xff0c;尤其是在網絡環境受限或需要在多臺機器上部署相同環境時。以下是詳細的步驟&#xff0c;幫助您創建一個包含所有依賴項的離線安裝包&#xff0c;并在后續環境中復用。 步驟 1&#xff1a;準備工具和環境 確保您有一臺可以訪問互…

為啥物聯網用MQTT?

前言 都說物聯網用MQTT&#xff0c;那分別使用Http和Mqtt發送“Hello”&#xff0c;比較一下就知道啦 HTTP HTTP請求報文由請求行、頭部字段和消息體組成。一個最簡單的HTTP POST請求如下&#xff1a; POST / HTTP/1.1 Host: example.com Content-Length: 5 Content-Type: …

操作系統 ------ 五種IO模型

阻塞IO&#xff1a;一個IO請求操作&#xff0c;準備階段和復制階段都會阻塞應用程序&#xff0c;直到操作完全完成 非阻塞IO&#xff1a;一個IO操作請求&#xff0c;先判斷準備階段是否完成&#xff0c;如果未完成立即返回&#xff0c;否則&#xff0c;進入復制階段&#xff0…

service和endpoints是如何關聯的?

在Kubernetes中&#xff0c;Service 和 Endpoints 是兩個密切關聯的對象&#xff0c;它們共同實現了服務發現和負載均衡的功能。以下是它們之間的關聯和工作原理&#xff1a; 1. Service 的定義 Service 是一種抽象&#xff0c;定義了一組邏輯上相關的 Pod&#xff0c;以及用…

程序化廣告行業(78/89):多因素交織下的行業剖析與展望

程序化廣告行業&#xff08;78/89&#xff09;&#xff1a;多因素交織下的行業剖析與展望 在程序化廣告這片充滿活力又不斷變化的領域&#xff0c;持續學習和知識共享是我們緊跟潮流、實現突破的關鍵。一直以來&#xff0c;我都渴望能與大家一同探索這個行業的奧秘&#xff0c…

數智化重構供應商管理

當供應鏈韌性成為核心競爭力&#xff0c;你的供應商管理還在 “摸著石頭過河” 嗎&#xff1f; 在傳統模式下&#xff0c;供應商管理高度依賴人工經驗與紙質流程&#xff1a; 入庫篩選如“大海撈針”&#xff1a;供應商資質審核停留在Excel表格比對&#xff0c;資質造假、歷史…

網絡互連與互聯網

1.在路由表中找不到目標網絡時使用默認路由&#xff0c;默認路由通常指本地網關的地址。 2.OSPF最主要的特征是使用分布式鏈路狀態協議&#xff0c;而RIP使用的是距離向量協議。 3.OSPF使用鏈路狀態公告LSA擴散路由信息 4.內部網關路由協議IGRP是一種動態距離矢量路由協議&a…

Raymarching Textures In Depth

本節課最主要的就是學會hlsl中使用紋理采樣 float4 color Texture2DSample(Texobj, TexobjSampler, uv); return color; 課程中的代碼&#xff08;沒有這張圖我就沒做&#xff09; 課程代碼產生深度的原因是uv偏移&#xff0c;黑色區域會不斷向左偏移&#xff0c;直到找到白色…

【MQTT-協議原理】

MQTT-協議原理 ■ MQTT-協議原理■ MQTT-服務器 稱為"消息代理"&#xff08;Broker&#xff09;■ MQTT協議中的訂閱、主題、會話■ 一、訂閱&#xff08;Subscription&#xff09;■ 二、會話&#xff08;Session&#xff09;■ 三、主題名&#xff08;Topic Name&a…

docker容器安裝的可道云掛接宿主機的硬盤目錄:解決群暉 威聯通 飛牛云等nas的硬盤掛接問題

基于Docker部署可道云&#xff08;KodCloud&#xff09;時&#xff0c;通過掛載宿主機其他磁盤目錄可實現高效、安全的數據管理。具體而言&#xff0c;使用綁定掛載&#xff08;Bind Mounts&#xff09;將宿主機目錄&#xff08;如/data/disk2&#xff09;映射到容器內的可道云…

go語言內存泄漏的常見形式

go語言內存泄漏 子字符串導致的內存泄漏 使用自動垃圾回收的語言進行編程時&#xff0c;通常我們無需擔心內存泄漏的問題&#xff0c;因為運行時會定期回收未使用的內存。但是如果你以為這樣就完事大吉了&#xff0c;哪里就大錯特措了。 因為&#xff0c;雖然go中并未對字符串…

es6學習02-let命令和const命令

一、let命令 1.let塊級作用域&#xff1a; let關鍵字 VS var關鍵字 2.for循環計數器很適合let命令 var&#xff1a;整個for循環中一直都是同一個i在做1&#xff0c;最后輸出的就是10&#xff1b; let&#xff1a;每循環一次都是多一個i的賦值&#xff0c;最后輸出是可以調出…

MySQL深分頁問題

在項目中有一個數據導出的需求&#xff0c;原來的實現方式也比較簡單&#xff0c;根據查詢條件分頁查所有的數據&#xff0c;然后轉成csv的格式一行一行寫進文件存儲中。 實際上線之后&#xff0c;發現出現了慢查詢&#xff0c;具體的sql如下&#xff1a; select * from tabl…

前端面試寶典---創建對象的配置

Object.create 對整個對象的多個屬性值進行配置 創建對象 不可更改屬性值 // 創建對象 不可更改屬性值 let obj Object.create({}, {name: {value: lisi,writable: false,},age: {value: 20,writable: true,} })console.log(初始化obj, obj) obj.name wangwu console.log(…

數據結構:C語言版嚴蔚敏和解析介紹,附pdf

《數據結構&#xff1a;C語言版&#xff08;第2版&#xff09;》嚴蔚敏李冬梅吳偉民.pdf 《數據結構&#xff1a;C語言版》嚴蔚敏&#xff0c;李冬梅.pdf 《數據結構C語言第2版習題解析與實驗指導》李冬梅.pdf 「《數據結構&#xff1a;C語言版&#xff08;第2版 &#xff09;》…

深入理解 v-for 指令及其使用方法

在 Vue.js 中&#xff0c;v-for 是用于渲染列表的核心指令&#xff0c;它允許你通過循環渲染數據源中的每一項。通過 v-for&#xff0c;你可以輕松地將數組、對象或其他可迭代的數據渲染成 HTML 元素。本文將詳細介紹 v-for 的基本用法、常見的應用場景、最佳實踐及性能優化&am…

VIRT, RES,SHR之間的關系

VIRT、RES 和 SHR 是進程內存使用的三個關鍵指標&#xff0c;它們之間的關系反映了進程的內存分配和使用情況。以下是它們的定義和關系&#xff1a; VIRT&#xff08;虛擬內存&#xff09;&#xff1a;表示進程分配的虛擬內存總量&#xff0c;包括所有代碼、數據、共享庫、堆棧…

2025屆藍橋杯JavaB組個人題解(暫時不全,沒題目)

2025 屆藍橋杯 Java B 組題解 第一次參加藍橋杯&#xff0c;輸入輸出都用的BufferedReader和PrintWriter&#xff0c;怕輸入輸出不對或者內存超限&#xff0c;也怕出現小錯誤運行不了的&#xff0c;比如Main打成Mian什么的&#xff0c;但還是希望能拿省一&#xff0c;這里給出自…

在Vue項目的引入meting-js音樂播放器插件

開源項目&#xff1a;https://github.com/swzaaaaaaa/NBlog 1、開源項目中音樂播放插件的使用流程 步驟1&#xff1a;下載meting-js相關文件 在MetingJS官方倉庫或其他可靠的CDN獲取meting-js的JavaScript文件以及相關依賴&#xff08;如APlayer的文件&#xff09;。將它們下…