欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

通俗易懂的C語言快速排序和歸并排序的時(shí)間復(fù)雜度分析

 更新時(shí)間:2023年01月26日 10:03:40   作者:特務(wù)依昂  
這篇文章主要為大家通俗易懂的講解了C語言快速排序和歸并排序的時(shí)間復(fù)雜度分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

快速排序和歸并排序的時(shí)間復(fù)雜度分析——通俗易懂

今天面試的時(shí)候,被問到歸并排序的時(shí)間復(fù)雜度,這個(gè)大家都知道是O(nlogn),但是面試官又繼續(xù)問,怎么推導(dǎo)出來的。這我就有點(diǎn)懵了,因?yàn)橹按_實(shí)沒有去真正理解這個(gè)時(shí)間復(fù)雜度是如何得出的,于是就隨便答了一波(理解了之后,發(fā)現(xiàn)面試的時(shí)候答錯(cuò)了......)。

歸并排序和快速排序,是算法中,非常重要的兩個(gè)知識(shí)點(diǎn),同時(shí)也是在面試中被問的非常頻繁的內(nèi)容,我明知如此,卻沒有徹底理解,真是太不應(yīng)該了。所以,今天這篇博客就來分析一下這兩種排序算法的時(shí)間復(fù)雜度是如何得出的。我查了許多篇博客,很多都是通過公式進(jìn)行分析,十分難理解,下面我就結(jié)合自己的理解,使用通俗易懂的方式進(jìn)行描述(為了好理解,可能會(huì)有些啰嗦)。

歸并排序的時(shí)間復(fù)雜度分析

了解歸并排序的應(yīng)該都知道,歸并排序的時(shí)間復(fù)雜度是O(nlogn),且這個(gè)時(shí)間復(fù)雜度是穩(wěn)定的,不隨需要排序的序列不同而產(chǎn)生波動(dòng)。那這個(gè)時(shí)間復(fù)雜度是如何得來的呢?我們可以這樣分析,假設(shè)我們需要對(duì)一個(gè)包含n個(gè)數(shù)的序列使用歸并排序,并且使用的是遞歸的實(shí)現(xiàn)方式,那么過程如下:

  • 遞歸的第一層,將n個(gè)數(shù)劃分為2個(gè)子區(qū)間,每個(gè)子區(qū)間的數(shù)字個(gè)數(shù)為n/2;
  • 遞歸的第二層,將n個(gè)數(shù)劃分為4個(gè)子區(qū)間,每個(gè)子區(qū)間的數(shù)字個(gè)數(shù)為n/4
  • 遞歸的第三層,將n個(gè)數(shù)劃分為8個(gè)子區(qū)間,每個(gè)子區(qū)間的數(shù)字個(gè)數(shù)為n/8;

......

  • 遞歸的第logn層,將n個(gè)數(shù)劃分為n個(gè)子區(qū)間,每個(gè)子區(qū)間的數(shù)字個(gè)數(shù)為1

我們知道,歸并排序的過程中,需要對(duì)當(dāng)前區(qū)間進(jìn)行對(duì)半劃分,直到區(qū)間的長(zhǎng)度為1。也就是說,每一層的子區(qū)間,長(zhǎng)度都是上一層的1/2。這也就意味著,當(dāng)劃分到第logn層的時(shí)候,子區(qū)間的長(zhǎng)度就是1了。而歸并排序的merge操作,則是從最底層開始(子區(qū)間為1的層),對(duì)相鄰的兩個(gè)子區(qū)間進(jìn)行合并,過程如下:

  • 在第logn層(最底層),每個(gè)子區(qū)間的長(zhǎng)度為1,共n個(gè)子區(qū)間,每相鄰兩個(gè)子區(qū)間進(jìn)行合并,總共合并n/2次。n個(gè)數(shù)字都會(huì)被遍歷一次,所有這一層的總時(shí)間復(fù)雜度為O(n);

......

  • 在第二層,每個(gè)子區(qū)間長(zhǎng)度為n/4,總共有4個(gè)子區(qū)間,每相鄰兩個(gè)子區(qū)間進(jìn)行合并,總共合并2次。n個(gè)數(shù)字都會(huì)被遍歷一次,所以這一層的總時(shí)間復(fù)雜度為O(n);
  • 在第一層,每個(gè)子區(qū)間長(zhǎng)度為n/2,總共有2個(gè)子區(qū)間,只需要合并一次。n個(gè)數(shù)字都會(huì)被遍歷一次,所以這一層的總時(shí)間復(fù)雜度為O(n);

通過上面的過程我們可以發(fā)現(xiàn),對(duì)于每一層來說,在合并所有子區(qū)間的過程中,n個(gè)元素都會(huì)被 操作一次,所以每一層的時(shí)間復(fù)雜度都是O(n)。而之前我們說過,歸并排序劃分子區(qū)間,將子區(qū)間劃分為只剩1個(gè)元素,需要?jiǎng)澐?code>logn次。每一層的時(shí)間復(fù)雜度為O(n),共有l(wèi)ogn層,所以歸并排序的時(shí)間復(fù)雜度就是O(nlogn) 。

上面的描述算是非常詳細(xì)了,應(yīng)該不會(huì)太難理解。如果上面的過程還是不太理解,那么我們通過另外一種更直觀的方式進(jìn)行分析。上面描述的是遞歸的過程,下面我們通過非遞歸(迭代)方式實(shí)現(xiàn)的歸并排序,再來分析一波,這種方式更加直觀(為什么不直接通過非遞歸的方式描述,而是先通過遞歸的方式分析,是因?yàn)樯厦娴倪^程也可以用來分析快速排序)。下面是通過非遞歸方式實(shí)現(xiàn)的歸并排序代碼,其中有兩處分析時(shí)間復(fù)雜度的關(guān)鍵點(diǎn),我標(biāo)注出來了(重點(diǎn)關(guān)注注釋):

**

/**
 * 此方法用來定義子區(qū)間大小,子區(qū)間大小從1->2->4->8 ... ->n/2
 * 可以近似地認(rèn)為進(jìn)行了logn次
 */
public static void merge(int[] arr) {
    // 關(guān)鍵點(diǎn)1:劃分子區(qū)間,每一次的子區(qū)間長(zhǎng)度是上一次的兩倍,所以這個(gè)循環(huán)需要執(zhí)行l(wèi)ogn次
    for(int i = 1;i<arr.length;i *= 2){
        // 關(guān)鍵點(diǎn)2:此方法每次執(zhí)行的時(shí)間復(fù)雜度為O(n),具體看下方
        mergeSort(arr,i);
    }
}
/**
 * 以下方法,每次執(zhí)行的時(shí)間復(fù)雜度都是O(n),
 * 因?yàn)樾枰獙rr數(shù)組的每gap個(gè)數(shù)子,作為一個(gè)子區(qū)間,
 * 然后對(duì)相鄰的兩個(gè)子區(qū)間執(zhí)行歸并排序的merge操作,
 * 所以在這個(gè)方法中,arr數(shù)組中的每一個(gè)數(shù)都會(huì)在merge操作中,
 * 被處理一次,所以下面這個(gè)方法的時(shí)間復(fù)雜度為O(n)
 */
public static void mergeSort(int[] arr, int gap) {
    int[] tmp = new int[arr.length];
    int index = 0;
    int start1 = 0;
    int end1 = start1 + gap - 1;
    int start2 = end1 + 1;
    int end2 = (start2 + gap - 1)>=arr.length?arr.length-1:start2+gap-1;
    while(start2<arr.length){
        while(start1<=end1&&start2<=end2){
            if(arr[start1]<arr[start2]){
                tmp[index++] = arr[start1++];
            }else{
                tmp[index++] = arr[start2++];
            }
        }
        while(start1<=end1){
            tmp[index++] = arr[start1++];
        }
        while(start2<=end2){
            tmp[index++] = arr[start2++];
        }
        start1 = end2+1;
        end1 = start1 + gap - 1;
        start2 = end1 + 1;
        end2 = (start2 + gap - 1)>=arr.length?arr.length-1:start2+gap-1;
    }
    while(start1<arr.length){
        tmp[index++] = arr[start1++];
    }
    for(int j = 0;j<tmp.length;j++){
        arr[j] = tmp[j];
    }
}

上面的代碼,merge方法中的循環(huán)需要循環(huán)logn次,每次循環(huán)都調(diào)用一次mergeSort方法,mergeSort方法的時(shí)間復(fù)雜度為O(n),所以很容易得出歸并排序的時(shí)間復(fù)雜度為O(nlogn)

快速排序的時(shí)間復(fù)雜度

了解快速排序的應(yīng)該知道,快速排序的時(shí)間復(fù)雜度在O(nlogn)~ O(n^2)之間,下面我就來分別分析這兩種情況:

(一)快速排序的最好情況O(nlogn)

這種情況下,其實(shí)和上面通過遞歸分析的歸并排序很類似,理解了歸并排序的時(shí)間復(fù)雜度分析,那這里應(yīng)該也很好理解??焖倥判虻膶?shí)現(xiàn)方式,就是在當(dāng)前區(qū)間中選擇一個(gè)軸,區(qū)間中所有比軸小的數(shù)都需要放到軸的左邊,而比軸大的數(shù)則放到軸的右邊。在理想的情況下,我們選取的軸剛好就是這個(gè)區(qū)間的中位數(shù)。也就是說,在操作之后,正好將區(qū)間分成了數(shù)字個(gè)數(shù)相等的左右兩個(gè)子區(qū)間。此時(shí)就和歸并排序基本一致了:

  • 遞歸的第一層,n個(gè)數(shù)被劃分為2個(gè)子區(qū)間,每個(gè)子區(qū)間的數(shù)字個(gè)數(shù)為n/2
  • 遞歸的第二層,n個(gè)數(shù)被劃分為4個(gè)子區(qū)間,每個(gè)子區(qū)間的數(shù)字個(gè)數(shù)為n/4;
  • 遞歸的第三層,n個(gè)數(shù)被劃分為8個(gè)子區(qū)間,每個(gè)子區(qū)間的數(shù)字個(gè)數(shù)為n/8;

......

  • 遞歸的第logn層,n個(gè)數(shù)被劃分為n個(gè)子區(qū)間,每個(gè)子區(qū)間的數(shù)字個(gè)數(shù)為1

以上過程與歸并排序基本一致,而區(qū)別就是,歸并排序是從最后一層開始進(jìn)行merge操作,自底向上;而快速排序則是從第一層開始,交換區(qū)間中數(shù)字的位置,也就是自頂向下。但是,merge操作和快速排序的調(diào)換位置操作,時(shí)間復(fù)雜度是一樣的,對(duì)于每一個(gè)區(qū)間,處理的時(shí)候,都需要遍歷一次區(qū)間中的每一個(gè)元素。這也就意味著,快速排序和歸并排序一樣,每一層的總時(shí)間復(fù)雜度都是O(n),因?yàn)樾枰獙?duì)每一個(gè)元素遍歷一次。而且在最好的情況下,同樣也是有logn層,所以快速排序最好的時(shí)間復(fù)雜度為O(nlogn)。

快速排序的最壞情況O(n^2)

下面我們?cè)賮碚f一說快速排序的最壞情況,這種情況就比較好理解了。什么是快速排序的最壞情況,那就是,對(duì)于每一個(gè)區(qū)間,我們?cè)谔幚淼臅r(shí)候,選取的軸剛好就是這個(gè)區(qū)間的最大值或者最小值。比如我們需要對(duì)n個(gè)數(shù)排序,而每一次進(jìn)行處理的時(shí)候,選取的軸剛好都是區(qū)間的最小值。于是第一次操作,在經(jīng)過調(diào)換元素順序的操作后,最小值被放在了第一個(gè)位置,剩余n-1個(gè)數(shù)占據(jù)了2到n個(gè)位置;第二次操作,處理剩下的n-1個(gè)元素,又將這個(gè)子區(qū)間的最小值放在了當(dāng)前區(qū)間的第1個(gè)位置,以此類推......每次操作,都只能將最小值放到第一個(gè)位置,而剩下的元素,則沒有任何變化。所以對(duì)于n個(gè)數(shù)來說,需要操作n次,才能為n個(gè)數(shù)排好序。而每一次操作都需要遍歷一次剩下的所有元素,這個(gè)操作的時(shí)間復(fù)雜度是O(n),所以總時(shí)間復(fù)雜度為O(n^2)

其實(shí)上面的過程,我們可以換一個(gè)角度理解:每次操作,找出最小值放到剩余區(qū)間的第一個(gè)位置,這不就是選擇排序的實(shí)現(xiàn)方式嗎?而選擇排序的時(shí)間復(fù)雜度就是O(n^2),所以上面的過程也就O(n^2)

總結(jié)

以上內(nèi)容,就是我基于自己的理解,對(duì)快速排序和歸并排序時(shí)間復(fù)雜度的分析。為了更好理解,我的描述都盡可能的詳細(xì),所以可能會(huì)有點(diǎn)啰嗦,但是我認(rèn)為還是很通俗易懂的。希望這篇博客能夠?yàn)橹皩?duì)這兩種排序算法理解不是特別清晰的人提供幫助,同時(shí),若上面的內(nèi)容存在錯(cuò)誤或不足,歡迎指正或補(bǔ)充。

以上就是通俗易懂的C語言快速排序和歸并排序的時(shí)間復(fù)雜度分析的詳細(xì)內(nèi)容,更多關(guān)于C語言排序時(shí)間復(fù)雜度的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • C語言讀取和存儲(chǔ)bmp格式圖片

    C語言讀取和存儲(chǔ)bmp格式圖片

    這篇文章主要為大家詳細(xì)介紹了C語言讀取和存儲(chǔ)bmp格式圖片,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-10-10
  • C++面試八股文之位運(yùn)算問題詳解

    C++面試八股文之位運(yùn)算問題詳解

    這篇文章主要為大家介紹了C++面試八股文之位運(yùn)算的問題解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-06-06
  • Matlab實(shí)現(xiàn)統(tǒng)計(jì)集合中各元素出現(xiàn)次數(shù)的示例代碼

    Matlab實(shí)現(xiàn)統(tǒng)計(jì)集合中各元素出現(xiàn)次數(shù)的示例代碼

    統(tǒng)計(jì)數(shù)組中各個(gè)元素?cái)?shù)量是一個(gè)很常用的功能,本文主要為大家介紹了如何利用Matlab優(yōu)雅的統(tǒng)計(jì)集合中各元素出現(xiàn)的次數(shù),感興趣的可以了解一下
    2022-05-05
  • C語言標(biāo)準(zhǔn)時(shí)間與秒單位相互轉(zhuǎn)換

    C語言標(biāo)準(zhǔn)時(shí)間與秒單位相互轉(zhuǎn)換

    這篇文章主要介紹了C語言標(biāo)準(zhǔn)時(shí)間與秒單位相互轉(zhuǎn)換,秒單位與標(biāo)準(zhǔn)時(shí)間的轉(zhuǎn)換方式,這份代碼一般用在嵌入式單片機(jī)里比較多,比如:設(shè)置RTC時(shí)鐘的時(shí)間,從RTC里讀取秒單位時(shí)間后,需要轉(zhuǎn)換成標(biāo)準(zhǔn)時(shí)間顯示。下文分享需要的小伙伴可以參考一下
    2022-05-05
  • C++實(shí)現(xiàn)LeetCode(59.螺旋矩陣之二)

    C++實(shí)現(xiàn)LeetCode(59.螺旋矩陣之二)

    這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(59.螺旋矩陣之二),本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-07-07
  • QT中QStringListModel類的應(yīng)用介紹

    QT中QStringListModel類的應(yīng)用介紹

    QStringListModel是最簡(jiǎn)單的模型類,具備向視圖提供字符串?dāng)?shù)據(jù)的能力,本文主要介紹了QT中QStringListModel類的應(yīng)用介紹,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-01-01
  • C++ 復(fù)制控制之復(fù)制構(gòu)造函數(shù)的實(shí)現(xiàn)

    C++ 復(fù)制控制之復(fù)制構(gòu)造函數(shù)的實(shí)現(xiàn)

    所謂的“復(fù)制控制”即通過這三個(gè)成員函數(shù)控制對(duì)象復(fù)制的過程,本文主要介紹了C++ 復(fù)制控制之復(fù)制構(gòu)造函數(shù)的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下
    2023-11-11
  • C語言實(shí)現(xiàn)簡(jiǎn)單圖書管理系統(tǒng)

    C語言實(shí)現(xiàn)簡(jiǎn)單圖書管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)圖書管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • 最新評(píng)論