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

c語(yǔ)言排序之歸并排序(遞歸和非遞歸)

 更新時(shí)間:2022年04月19日 09:19:09   作者:sndapk  
這篇文章主要介紹了?c語(yǔ)言排序之歸并排序(遞歸和非遞歸),歸并就是把兩個(gè)或多個(gè)序列合并,本文主要介紹二路歸并,下文相關(guān)資料需要的小伙伴可以參考一下

遞歸代碼流程

歸并就是把兩個(gè)或多個(gè)序列合并,這里只介紹二路歸并,就是不斷的把序列分為2組,直到每個(gè)組有一個(gè)元素為止,然后再比較合并,直到合為1個(gè)序列,完成。

非遞歸代碼流程

與遞歸不斷分解數(shù)組相反,非遞歸直接從長(zhǎng)度為1的子序列開(kāi)始合并,直到全并為1個(gè)整個(gè)序列,復(fù)用了merge函數(shù)

兩者比較

代碼用非遞歸的方式效率更高一些:

? 空間復(fù)雜度:從O(log2n)變?yōu)?個(gè)臨時(shí)數(shù)組O(n)

? 時(shí)間復(fù)雜度:少了遞歸的時(shí)間

時(shí)間復(fù)雜度

O(nlogn)

代碼(遞歸)

#include <stdio.h>
#include <stdbool.h>

#define MAXSIZE 9

typedef struct {
    int r[MAXSIZE+1]; // first index used as tmp, not real data
    int len;
}SqList;

void swap(SqList *L, int i, int j) {
    int tmp = L->r[i];
    L->r[i] = L->r[j];
    L->r[j] = tmp;
}

void merge(int sr[], int tr[], int s, int m, int t) {
    // 本函數(shù)的任務(wù)就是比對(duì)sr中兩個(gè)分組(s..m, m+1..t)的元素大小并歸并到tr
    int j,k,l;

    j = m + 1; // 第2分組的起始位置
    k = s; // k用于tr數(shù)組中的游標(biāo)與sr中的起始位置對(duì)應(yīng)起來(lái)
    while (s<=m && j<=t) {
        if (sr[s] < sr[j]) {
            tr[k++] = sr[s++];
        }
        else {
            tr[k++] = sr[j++];
        }
    }

    // 只要是合并,就肯定至少是2個(gè)序列合并,肯定會(huì)在比對(duì)后剩下1個(gè)未消耗完元素的序列分組
    while (s<=m) {
        tr[k++] = sr[s++];
    }

    while (j<=t) {
        tr[k++] = sr[j++];
    }
}

void msort(int sr[], int tr[], int s, int t) {
    /*
     * 把sr進(jìn)行歸并排序并有序保存到(歸并到)tr中
     */
    int m;
    int tmpr[MAXSIZE+1]; // 每層遞歸的臨時(shí)數(shù)組,存放本次被調(diào)用時(shí)s到t歸并后的下標(biāo)值(位置與首次傳入的L->r相同)

    if (s == t) {
        tr[s] = sr[s]; // 歸并的思想,1個(gè)元素的分組為有序
    }
    else { // 不是1個(gè)元素的分組,繼續(xù)分組
        m = (s+t)/2;
        msort(sr, tmpr, s, m);
        msort(sr, tmpr, m+1, t);
        // 合并tmpr到tr完成本層的排序任務(wù)
        merge(tmpr, tr, s, m, t);
    }
}

void merge_sort(SqList *L) {
    msort(L->r, L->r, 1, L->len); // 因?yàn)樵趍sort中第1個(gè)參數(shù)sr數(shù)組只是讀取,所以這里這樣傳遞沒(méi)有問(wèn)題
}


int main(void) {
    SqList list = {
        {999,50,10,90,30,70,40,80,60,20},
        MAXSIZE
    };

    merge_sort(&list);
    printf("after merge_sort:\n");
    for (int i=0; i<=MAXSIZE; i++) {
        printf("index: %d, value: %d\n",i,list.r[i]);
    }
    return 0;
}

output

?  c gcc sort_merge.c&& ./a.out
after merge_sort:
index: 0, value: 999
index: 1, value: 10
index: 2, value: 20
index: 3, value: 30
index: 4, value: 40
index: 5, value: 50
index: 6, value: 60
index: 7, value: 70
index: 8, value: 80
index: 9, value: 90
 

代碼(非遞歸)

#include <stdio.h>
#include <stdbool.h>
#include <stdlib.h>

#define MAXSIZE 9

typedef struct {
    int r[MAXSIZE+1]; // first index used as tmp, not real data
    int len;
}SqList;

void merge(int sr[], int tr[], int s, int m, int t) {
    // 本函數(shù)的任務(wù)就是比對(duì)sr中兩個(gè)分組(s..m, m+1..t)的元素大小并歸并到tr
    int j,k,l;

    j = m + 1; // 第2分組的起始位置
    k = s; // k用于tr數(shù)組中的游標(biāo)與sr中的起始位置對(duì)應(yīng)起來(lái)
    while (s<=m && j<=t) {
        if (sr[s] < sr[j]) {
            tr[k++] = sr[s++];
        }
        else {
            tr[k++] = sr[j++];
        }
    }

    // 只要是合并,就肯定至少是2個(gè)序列合并,肯定會(huì)在比對(duì)后剩下1個(gè)未消耗完元素的序列分組
    while (s<=m) {
        tr[k++] = sr[s++];
    }

    while (j<=t) {
        tr[k++] = sr[j++];
    }
}


void merge_pass(int sr[], int tr[], int k, int len) {
    int i=1; // 合并時(shí)的游標(biāo)
    while (i < len-2*k+1) { // 也就是每次循環(huán)后,當(dāng)前所剩余的是否還夠2個(gè)完整子序列
        merge(sr, tr, i, i+k-1, i+2*k-1); //合并本輪掃描到的2個(gè)子序列
        i+=2*k; // 賦值后的i為下一輪2個(gè)子序列的起始位置
    }

    // 下面是掃尾工作,**可能會(huì)**出現(xiàn)2種情況,a. 剩余1~2個(gè)子序列之間的情況, b. 剩余<=1個(gè)子序列的情況
    if (i < len-k+1) {
        merge(sr, tr, i, i+k-1, len);
    }
    else { // 這里加else也可以 如果上面i正好把序列消耗完,則循環(huán)不會(huì)執(zhí)行
        while (i<len) {
            tr[i] = sr[i];
            i++;
        }
    }
}

void merge_sort(SqList *L) {
    int *tr = (int *)malloc(L->len*sizeof(int));
    int k=1;
    /*
     * 循環(huán)k為序列長(zhǎng)度,與遞歸的方式相比,正好反過(guò)來(lái),非遞歸方式直接從序列為1開(kāi)始合并,直到序列不小于待排序的數(shù)組長(zhǎng)度為止
     * 每次循環(huán)都是子序列*4變長(zhǎng)的過(guò)程
     */ 
    while (k<L->len) {
        merge_pass(L->r, tr, k, L->len); // 序列1變2
        k++;
        merge_pass(tr, L->r, k, L->len); // 序列2變4
        k++;
    }
}


int main(void) {
    SqList list = {
        {999,50,10,90,30,70,40,80,60,20},
        MAXSIZE
    };

    merge_sort(&list);
    printf("after merge_sort2:\n");
    for (int i=0; i<=MAXSIZE; i++) {
        printf("index: %d, value: %d\n",i,list.r[i]);
    }
    return 0;
}

output

?  c gcc sort_merge_norecursive.c&& ./a.out
after merge_sort2:
index: 0, value: 999
index: 1, value: 10
index: 2, value: 20
index: 3, value: 30
index: 4, value: 40
index: 5, value: 50
index: 6, value: 60
index: 7, value: 70
index: 8, value: 80
index: 9, value: 90

到此這篇關(guān)于 c語(yǔ)言排序之歸并排序(遞歸和非遞歸)的文章就介紹到這了,更多相關(guān) c語(yǔ)言排序內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C語(yǔ)言 90后懷舊游戲超級(jí)瑪麗的實(shí)現(xiàn)流程

    C語(yǔ)言 90后懷舊游戲超級(jí)瑪麗的實(shí)現(xiàn)流程

    90后最風(fēng)靡的游戲是什么?第一個(gè)聯(lián)想到的肯定是插卡游戲機(jī)或者VCD加光盤運(yùn)行在電視機(jī)上的超級(jí)瑪麗了,它的經(jīng)典絕對(duì)可以排在第一位,長(zhǎng)大后的我們今天來(lái)用C語(yǔ)言重溫一下
    2021-11-11
  • C語(yǔ)言PlaySound函數(shù)使用方法

    C語(yǔ)言PlaySound函數(shù)使用方法

    這篇文章介紹了C語(yǔ)言PlaySound函數(shù)的使用方法,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-12-12
  • C++類的空指針調(diào)用成員函數(shù)的代碼

    C++類的空指針調(diào)用成員函數(shù)的代碼

    這篇文章主要介紹了C++類的空指針調(diào)用成員函數(shù),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-07-07
  • c語(yǔ)言指針數(shù)組的具體使用

    c語(yǔ)言指針數(shù)組的具體使用

    指針數(shù)組就是存放指針變量的數(shù)組,指針數(shù)組的本質(zhì)是數(shù)組,而非指針,本文主要介紹了c語(yǔ)言指針數(shù)組的具體使用,具有一定的參考價(jià)值,感興趣的可以了解一下
    2023-12-12
  • 使用C語(yǔ)言編寫圣誕表白程序

    使用C語(yǔ)言編寫圣誕表白程序

    圣誕節(jié)快到了,讓我們用C語(yǔ)言制作一個(gè)圣誕表白程序吧,下面通過(guò)本文學(xué)習(xí)下實(shí)現(xiàn)代碼
    2016-12-12
  • 詳解C標(biāo)準(zhǔn)庫(kù)堆內(nèi)存函數(shù)

    詳解C標(biāo)準(zhǔn)庫(kù)堆內(nèi)存函數(shù)

    在C/C++語(yǔ)言中,我們知道內(nèi)存分為這幾種:程序全局變量?jī)?nèi)存、棧內(nèi)存、堆內(nèi)存。其中堆內(nèi)存就是通過(guò)malloc(new)來(lái)分配的內(nèi)存,本文我們來(lái)探討一下C標(biāo)準(zhǔn)庫(kù)堆內(nèi)存函數(shù)。
    2021-06-06
  • C++連接mysql數(shù)據(jù)庫(kù)(改進(jìn)版)

    C++連接mysql數(shù)據(jù)庫(kù)(改進(jìn)版)

    C++是大家都非常熟悉的,也是大家平時(shí)辦公中經(jīng)常會(huì)用到的,下面這篇文章主要給大家介紹了關(guān)于C++連接mysql數(shù)據(jù)庫(kù)的相關(guān)資料,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-12-12
  • C++實(shí)現(xiàn)LeetCode(648.替換單詞)

    C++實(shí)現(xiàn)LeetCode(648.替換單詞)

    這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(648.替換單詞),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-08-08
  • C++的array和&array有什么區(qū)別

    C++的array和&array有什么區(qū)別

    本文主要介紹了C++的array和&array有什么區(qū)別,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2022-08-08
  • C++圖文并茂講解繼承

    C++圖文并茂講解繼承

    繼承是C++面向?qū)ο缶幊讨械囊婚T。繼承是子類繼承父類的特征和行為,或者是繼承父類得方法,使的子類具有父類得的特性和行為。重寫是子類對(duì)父類的允許訪問(wèn)的方法實(shí)行的過(guò)程進(jìn)行重新編寫,返回值和形參都不能改變。就是對(duì)原本的父類進(jìn)行重新編寫,但是外部接口不能被重寫
    2022-05-05

最新評(píng)論