標籤:lzw演算法 c++
LZW編碼通過建立一個字串表,用較短的代碼來表示較長的字串來實現壓縮。 LZW壓縮演算法是Unisys的專利,有效期間到2003年,所以相關演算法大多也已到期。
本代碼僅僅完成了LZW的編碼與解碼演算法功能,相對網上找到的很多代碼而言較為簡(cai)單(bi),瞭解struct && 會遞迴即可,算是優點吧。
#include <stdio.h>#include <algorithm>#include <math.h>#include <iostream>#include <stdlib.h>#include <string.h>using namespace std;struct Node{int pre;//首碼單詞對應碼字char c;//當前字元}KDA[65535];int cnt,P,Q;char W,V;void Init(){//前256個字典由對應ASCII碼產生for(int i = 0;i < 256; i ++){KDA[i].pre = -1;KDA[i].c = i;}cnt = 256;P = -1;}void Out(int x){ //遞迴輸出碼字對應單詞,首位另存用於建立字典 if(KDA[x].pre != -1){ Out(KDA[x].pre); } else { V = KDA[x].c; } printf("%c",KDA[x].c);}void Search(){ int flag = 0;for(int i = 0;i < cnt;i ++){if(KDA[i].pre == P && KDA[i].c == W){//字典已存在則更新首碼對應碼字P = i;flag = 1; }}if(!flag){ //不存在則擴充字典,輸出首碼對應碼字並更新首碼單詞 KDA[cnt].pre = P; KDA[cnt].c = W; printf("%03X ",P); P = (int)W; cnt ++; }}void Research(){ Out(Q); if(P != -1){ //如果前一位碼字不為空白,則將前一位對應單詞作為首碼與本單詞第一位合并作為新單詞加入字典 KDA[cnt].pre = P; KDA[cnt].c = V; cnt ++; }}void Compress(){<span style="white-space:pre"></span>//編碼過程Init();freopen("LZWin.txt","r",stdin);freopen("LZWch.txt","w",stdout);while((W = getchar()) && W != EOF){Search();}printf("%03X\n",P);}void Decompress(){<span style="white-space:pre"></span>//解碼過程 Init(); freopen("LZWch.txt","r",stdin);freopen("LZWout.txt","w",stdout);while(scanf("%03X",&Q)!= EOF){Research();}}int main(){ Compress(); Decompress();return 0;}
【手打】LZW編碼的C/C++實現