自底向上的哈夫曼編碼

哈夫曼編碼原理:哈夫曼編碼原理
練習(xí)題目:哈夫曼編碼

其中第一個即是自底向上的硕舆,另外還有幾個練習(xí)題惋增,可以進行相應(yīng)練習(xí)补胚。

#include<cstdio>
#include<iostream>
#include<cmath>
#include<cstring>
#include<string>
#include<algorithm>


using namespace std;
//輸入數(shù)字個數(shù) 
const int maxn = 110;
//輸入的需編碼的數(shù)字?jǐn)?shù)組 
int w[maxn];

//哈夫曼樹節(jié)點定義
//每個節(jié)點包括其權(quán)值壹哺,父親節(jié)點指針以及左右孩子結(jié)點指針 
typedef struct{
    int wt;
    int parent, lchild, rchild;
}hftree;

/***************************************************************
/*從已有的節(jié)點中選出權(quán)值最小的兩個結(jié)點
/*參數(shù):hftree倾剿,根據(jù)節(jié)點權(quán)值建立后的哈夫曼樹鸳吸,n當(dāng)前已有節(jié)點個數(shù)
/*      a第一小的節(jié)點熏挎,b第二小的節(jié)點 
*****************************************************************/
void min_two(hftree* tree, int n, int &a, int &b){
    //m初始設(shè)置為一個很大的數(shù) 
    int m = 0x3fffffff;
    //在已有的節(jié)點中尋找兩個權(quán)值最小的節(jié)點 
    for(int i = 1; i <= n; i++){
        if(tree[i].parent == 0 && m > tree[i].wt){
            m = tree[i].wt;
            a = i;
        }
    }
    //尋找第二個 
    m = 0x3fffffff;
    for(int i = 1; i <= n; i++){
        if(tree[i].parent == 0 && m > tree[i].wt && i != a){
            m = tree[i].wt;
            b = i;
        }
    }
    //由于當(dāng)選出兩個最小的節(jié)點之后,我們將兩個節(jié)點所放的左右順序不同時
    //得到的編碼順序也是不同的晌砾,為了得到一個固定的編碼坎拐,同時為了能夠
    //AC題目,將兩個中更小的放在左邊
    /****若無此步得到的編碼將是不唯一的 ***/
    if(a > b)
        swap(a, b);
}
/*********************************************************************************
/*         哈夫曼樹的建立以及編碼過程
/* 根據(jù)輸入的n個權(quán)值初始化n個葉子節(jié)點,再依次選擇未合并的節(jié)點(包括新節(jié)點)中的最小
/*的兩個權(quán)值節(jié)點進行合并哼勇,直到只剩下一個未合并的節(jié)點即根結(jié)點時結(jié)束都伪,
/* 然后則進行自底向上的編碼,每次都從某一個葉子節(jié)點出發(fā)积担,如果是其父節(jié)點的左孩子則添0 
/*如果是右孩子則添1陨晶,直到碰到根結(jié)點結(jié)束 
/********************************************************************************/
void HuffmanCode(hftree* &t, char** &code, int w[], int n){
    //當(dāng)輸入的n為1或者0時無需編碼 
    if(n <= 1)
        return ;
    //由于每次在找兩個節(jié)點并合成時,都會生成一個新的節(jié)點帝璧,而n個結(jié)點則會生成n-1個
    //而生成的節(jié)點是哈夫曼樹中的非葉子節(jié)點先誉,因此需要定義足夠的數(shù)組空間 
    int m = 2 * n - 1;
    //根據(jù)得到的節(jié)點個數(shù)為哈夫曼樹分配空間,0號空間不使用 
    t = (hftree *)malloc((m+1)*sizeof(hftree));
    //使用初始的n個數(shù)初始化哈夫曼樹的烁,相當(dāng)于初始化葉子節(jié)點 
    for(int i = 1; i <= n; i++){
        t[i].wt = w[i];
        t[i].parent = t[i].lchild = t[i].rchild = 0;
    }
    //對非葉子節(jié)點初始化信息褐耳,由于它們是之后生成的,因此沒有初始化權(quán)值信息 
    for(int i = n+1; i <= m; i++)
        t[i].parent = t[i].lchild = t[i].rchild = 0;
    //生成n-1個非葉子節(jié)點渴庆,每次都選擇已經(jīng)存在的且未合并的節(jié)點中的兩個最小節(jié)點
    //選出后將其合并铃芦,生成新的節(jié)點(此處當(dāng)parent為0時可以代表還未合并) 
    for(int i = n+1; i <= m; i++){
        int a, b;
        min_two(t, i-1, a, b);
        //對合并的節(jié)點進行信息修改,包括權(quán)值襟雷,父節(jié)點指向刃滓,左右孩子指向 
        t[i].wt = t[a].wt + t[b].wt;
        t[i].lchild = a;
        t[i].rchild = b;
        t[a].parent = t[b].parent = i;
    }
    
    //以下進行編碼操作
    //申請n個節(jié)點需要的編碼串的空間,由于一個編碼串可用一維指針表示
    //而n個時耸弄,便可用一個二維指針表示咧虎,它的每一維都對應(yīng)一個數(shù)的編碼 
    code = (char **)malloc((n+1)*sizeof(char *));
    //臨時存儲某個數(shù)的編碼串 
    char *cd = (char *)malloc(n*sizeof(char));
    //由于是自底向上的,所以進行逆向編碼叙赚,首先初始化末尾為字符串結(jié)束符 
    cd[n-1] = '\0';
    //對n個數(shù)進行編碼
    for(int i = 1; i <= n; i++){
        int index = n - 1;
        //從每一個i所在的葉子節(jié)點開始向上老客,如果當(dāng)前節(jié)點是其父節(jié)點的左孩子則添加一個0
        //否則添加1,當(dāng)回溯到根結(jié)點則i所在的葉子節(jié)點編碼完成 
        for(int c=i, f=t[i].parent; f != 0; c=f, f=t[f].parent){
            if(t[f].lchild == c)
                cd[--index] = '0';
            else
                cd[--index] = '1';
        }
        code[i] = (char*)malloc((n-index)*sizeof(char));
        //將i葉子節(jié)點的編碼串賦值到code中震叮,存儲起來 
        strcpy(code[i], cd+index);
    }
    //釋放空間 
    delete cd;
}


int main() {
    int n, a;
    while(scanf("%d", &n) != EOF){
        for(int i = 1; i <= n; i++)
            scanf("%d", &w[i]);
        hftree *ht;
        char** code;
        HuffmanCode(ht, code, w, n);
        for(int i = 1; i <= n; i++)
            printf("%s\n", code[i]);
        delete ht;
        delete code;
    }
    return 0;
}

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末胧砰,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子苇瓣,更是在濱河造成了極大的恐慌尉间,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,627評論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件击罪,死亡現(xiàn)場離奇詭異哲嘲,居然都是意外死亡,警方通過查閱死者的電腦和手機媳禁,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,180評論 3 399
  • 文/潘曉璐 我一進店門眠副,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人竣稽,你說我怎么就攤上這事囱怕』舻” “怎么了?”我有些...
    開封第一講書人閱讀 169,346評論 0 362
  • 文/不壞的土叔 我叫張陵娃弓,是天一觀的道長典格。 經(jīng)常有香客問我,道長台丛,這世上最難降的妖魔是什么耍缴? 我笑而不...
    開封第一講書人閱讀 60,097評論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮挽霉,結(jié)果婚禮上防嗡,老公的妹妹穿的比我還像新娘。我一直安慰自己炼吴,他們只是感情好本鸣,可當(dāng)我...
    茶點故事閱讀 69,100評論 6 398
  • 文/花漫 我一把揭開白布疫衩。 她就那樣靜靜地躺著硅蹦,像睡著了一般。 火紅的嫁衣襯著肌膚如雪闷煤。 梳的紋絲不亂的頭發(fā)上童芹,一...
    開封第一講書人閱讀 52,696評論 1 312
  • 那天,我揣著相機與錄音鲤拿,去河邊找鬼假褪。 笑死,一個胖子當(dāng)著我的面吹牛近顷,可吹牛的內(nèi)容都是我干的生音。 我是一名探鬼主播,決...
    沈念sama閱讀 41,165評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼窒升,長吁一口氣:“原來是場噩夢啊……” “哼缀遍!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起饱须,我...
    開封第一講書人閱讀 40,108評論 0 277
  • 序言:老撾萬榮一對情侶失蹤域醇,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后蓉媳,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體譬挚,經(jīng)...
    沈念sama閱讀 46,646評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,709評論 3 342
  • 正文 我和宋清朗相戀三年酪呻,在試婚紗的時候發(fā)現(xiàn)自己被綠了减宣。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,861評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡玩荠,死狀恐怖漆腌,靈堂內(nèi)的尸體忽然破棺而出丰歌,到底是詐尸還是另有隱情,我是刑警寧澤屉凯,帶...
    沈念sama閱讀 36,527評論 5 351
  • 正文 年R本政府宣布立帖,位于F島的核電站,受9級特大地震影響悠砚,放射性物質(zhì)發(fā)生泄漏晓勇。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 42,196評論 3 336
  • 文/蒙蒙 一灌旧、第九天 我趴在偏房一處隱蔽的房頂上張望绑咱。 院中可真熱鬧,春花似錦枢泰、人聲如沸描融。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,698評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽窿克。三九已至,卻和暖如春毛甲,著一層夾襖步出監(jiān)牢的瞬間年叮,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,804評論 1 274
  • 我被黑心中介騙來泰國打工玻募, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留只损,地道東北人。 一個月前我還...
    沈念sama閱讀 49,287評論 3 379
  • 正文 我出身青樓七咧,卻偏偏與公主長得像跃惫,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子艾栋,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,860評論 2 361

推薦閱讀更多精彩內(nèi)容