LZ77簡介
Ziv和Lempel于1977年發(fā)表題為“順序數(shù)據(jù)壓縮的一個通用算法(A Universal Algorithm for Sequential Data Compression )”的論文,論文中描述的算法被后人稱為LZ77算法。值得說的是,LZ77嚴格意義上來說不是一種算法,而是一種編碼理論。同Huffman編碼一樣,只定義了原理,并沒有定義如何實現(xiàn)。基于這種理論來實現(xiàn)的算法才稱為LZ77算法,或者人們更愿意稱為LZ77變種。實際上這類算法已經(jīng)有很多了,比如LZSS、LZB、LZH等。至今,幾乎我們?nèi)粘J褂玫乃型ㄓ脡嚎s工具,象ARJ,PKZip,WinZip,LHArc,RAR,GZip,ACE,ZOO,TurboZip,Compress,JAR??甚至許多硬件如網(wǎng)絡(luò)設(shè)備中內(nèi)置的壓縮算法,無一例外,都可以最終歸結(jié)為這兩個以色列人的杰出貢獻。
LZ77是一種基于字典的算法,它將長字符串(也稱為短語)編碼成短小的標記,用小標記代替字典中的短語,從而達到壓縮的目的。也就是說,它通過用小的標記來代替數(shù)據(jù)中多次重復出現(xiàn)的長串方法來壓縮數(shù)據(jù)。其處理的符號不一定是文本字符,可以是任意大小的符號。
短語字典的維護
不同的基于字典的算法使用不同的方法來維護它們的字典。LZ77使用的是一個前向緩沖區(qū)和一個滑動窗口。
LZ77首先將一部分數(shù)據(jù)載入前向緩沖區(qū)。為了便于理解前向緩沖區(qū)如何存儲短語并形成字典,我們將緩沖區(qū)描繪成S1,…,Sn的字符序列,Pb是由字符組成的短語集合。從字符序列S1,…,Sn,組成n個短語,定義如下:
Pb= {(S1),(S1,S2),…,(S1,…,Sn)}
例如,如果前向緩沖區(qū)包含字符(A,B,D),那么緩沖區(qū)中的短語為{(A),(A,B),(A,B,D)}。
一旦數(shù)據(jù)中的短語通過前向緩沖區(qū),那么它將移動到滑動窗口中,并變成字典的一部分。為理解短語是如何在滑動窗口中表示的,首先,把窗口想象成S1,…,Sm的字符序列,且Pw是由這些字符組成的短語集合。從序列S1,…,Sm產(chǎn)生短語數(shù)據(jù)集合的過程如下:
Pw= {P1,P2,…,Pm},其中Pi= {(Si),(Si,Si+1),…,(Si,Si+1,…,Sm)}
例如,如果滑動窗口中包含符號(A,B,C),那么窗口和字典中的短語為{(A),(A,B),(A,B,C),(B),(B,C),(C)}。
LZ77算法的主要思想就是在前向緩沖區(qū)中不斷尋找能夠與字典中短語匹配的最長短語。以上面描述的前向緩沖區(qū)和滑動窗口為例,其最長的匹配短語為(A,B)。
壓縮和解壓縮數(shù)據(jù)
前向緩沖區(qū)和滑動窗口之間的匹配有兩種情況:要么找到一個匹配短語,要么找不到匹配的短語。當找到最長的匹配時,將其編碼成短語標記。
短語標記包含三個部分:1、滑動窗口中的偏移量(從頭部到匹配開始的前一個字符);2、匹配中的符號個數(shù);3、匹配結(jié)束后,前向緩沖區(qū)中的第一個符號。
當沒有找到匹配時,將未匹配的符號編碼成符號標記。這個符號標記僅僅包含符號本身,沒有壓縮過程。事實上,我們將看到符號標記實際上比符號多一位,所以會出現(xiàn)輕微的擴展。
一旦把n個符號編碼并生成相應(yīng)的標記,就將這n個符號從滑動窗口的一端移出,并用前向緩沖區(qū)中同樣數(shù)量的符號來代替它們。然后,重新填充前向緩沖區(qū)。這個過程使滑動窗口中始終有最新的短語?;瑒哟翱诤颓跋蚓彌_區(qū)具體維護的短語數(shù)量由它們自身的容量決定。
下圖(1)展示了用LZ77算法壓縮字符串的過程,其中滑動窗口大小為8個字節(jié),前向緩沖區(qū)大小為4個字節(jié)。在實際中,滑動窗口典型的大小為4KB(4096字節(jié))。前向緩沖區(qū)大小通常小于100字節(jié)。
圖(1):使用LZ77算法對字符串ABABCBABABCAD進行壓縮
我們通過解碼標記和保持滑動窗口中符號的更新來解壓縮數(shù)據(jù),其過程類似于壓縮過程。當解碼每個標記時,將標記編碼成字符拷貝到滑動窗口中。每當遇到一個短語標記時,就在滑動窗口中查找相應(yīng)的偏移量,同時查找在那里發(fā)現(xiàn)的指定長度的短語。每當遇到一個符號標記時,就生成標記中保存的一個符號。下圖(2)展示了解壓縮圖(1)中數(shù)據(jù)的過程。
圖(2):使用LZ77算法對圖(1)中壓縮的字符串進行解壓縮
LZ77的效率
用LZ77算法壓縮的程度取決于很多因素,例如,選擇滑動窗口的大小,為前向緩沖區(qū)設(shè)置的大小,以及數(shù)據(jù)本身的熵。最終,壓縮的程度取決于能匹配的短語的數(shù)量和短語的長度。大多數(shù)情況下,LZ77比霍夫曼編碼有著更高的壓縮比,但是其壓縮過程相對較慢。
用LZ77算法壓縮數(shù)據(jù)是非常耗時的,國為要花很多時間尋找窗口中的匹配短語。然而在通常情況下,LZ77的解壓縮過程要比霍夫曼編碼的解壓縮過程耗時要少。LZ77的解壓縮過程非??焓且驗槊總€標記都明確地告訴我們在緩沖區(qū)中哪個位置可以讀取到所需要的符號。事實上,我們最終只從滑動窗口中讀取了與原始數(shù)據(jù)數(shù)量相等的符號而已。
LZ77的接口定義
lz77_compress
int lz77_compress(const unsigned char *original, unsigned char **compressed, int size);
返回值:如果數(shù)據(jù)壓縮成功,返回壓縮后數(shù)據(jù)的字節(jié)數(shù);否則返回-1;
描述: 用LZ77算法壓縮緩沖區(qū)original中的數(shù)據(jù),original包含size個字節(jié)的空間。壓縮后的數(shù)據(jù)存入緩沖區(qū)compressed中。lz77_compress需要調(diào)用malloc來動態(tài)的為compressed分配存儲空間,當這塊空間不再使用時,由調(diào)用者調(diào)用函數(shù)free來釋放空間。
復雜度:O(n),其中n是原始數(shù)據(jù)中符號的個數(shù)。
lz77_uncompress
int lz77_uncompress(const unsigned char *compressed, unsigned char **original);
返回值:如果解壓縮數(shù)據(jù)成功,返回恢復后數(shù)據(jù)的字節(jié)數(shù);否則返回-1;
描述: 用LZ77算法解壓縮緩沖區(qū)compressed中的數(shù)據(jù)。假定緩沖區(qū)包含的數(shù)據(jù)之前由lz77_compress壓縮。恢復后的數(shù)據(jù)存入緩沖區(qū)original中。lz77_uncompress函數(shù)調(diào)用malloc來動態(tài)的為original分配存儲空間。當這塊存儲空間不再使用時,由調(diào)用者調(diào)用函數(shù)free來釋放空間。
復雜度:O(n)其中n是原始數(shù)據(jù)中符號的個數(shù)。
LZ77的實現(xiàn)與分析
LZ77算法,通過一個滑動窗口將前向緩沖區(qū)中的短語編碼成相應(yīng)的標記,從而達到壓縮的目的。在解壓縮的過程中,將每個標記解碼成短語或符號本身。要做到這些,必須要不斷地更新窗口,這樣,在壓縮過程中的任何時刻,窗口都能按照規(guī)則進行編碼。在本節(jié)所有的示例中,原始數(shù)據(jù)中的一個符號占一個字節(jié)。
lz77_compress
lz77_compress操作使用LZ77算法來壓縮數(shù)據(jù)。首先,它將數(shù)據(jù)中的符號寫入壓縮數(shù)據(jù)的緩沖區(qū)中,并同時初始化滑動窗口和前向緩沖區(qū)。隨后,前向緩沖區(qū)將用來加載符號。
壓縮發(fā)生在一個循環(huán)中,循環(huán)會持續(xù)迭代直到處理完所有符號。使用ipos來保存原始數(shù)據(jù)中正在處理的當前字節(jié),并用opos來保存向壓縮數(shù)據(jù)緩沖區(qū)寫入的當前位。在循環(huán)的每次迭代中,調(diào)用compare_win來確定前向緩沖區(qū)與滑動窗口中匹配的最長短語。函數(shù)compare_win返回最長匹配串的長度。
當找到一個匹配串時,compare_win設(shè)置offset為滑動窗口中匹配串的位置,同時設(shè)置next為前向緩沖區(qū)中匹配串后一位的符號。在這種情況下,向壓縮數(shù)據(jù)中寫入一個短語標記(如圖3-a)。在本節(jié)展示的實現(xiàn)中,對于偏移量offset短語標記需要12位,這是因為滑動窗口的大小為4KB(4096字節(jié))。此時短語標志需要5位來表示長度,因為在一個32字節(jié)的前向緩沖區(qū)中,不會有匹配串超過這個長度。當沒有找到匹配串時,compare_win返回,并且設(shè)置next為前向緩沖區(qū)起始處未匹配的符號。在這種情況下,向壓縮數(shù)據(jù)中寫入一個符號(如圖3-b)。無論向壓縮數(shù)據(jù)中寫入的是一個短語還是一個符號,在實際寫入標記之前,都需要調(diào)用網(wǎng)絡(luò)函數(shù)htonl來轉(zhuǎn)換串,以保證標記是大端格式。這種格式是在實際壓縮數(shù)據(jù)和解壓縮數(shù)據(jù)時所要求的。
圖3:LZ77中的短語標記(A)和符號標記(B)的結(jié)構(gòu)
一旦將相應(yīng)的標記寫入壓縮數(shù)據(jù)的緩沖區(qū)中,就調(diào)整滑動窗口和前向緩沖區(qū)。要使數(shù)據(jù)通過滑動窗口,將數(shù)據(jù)從右邊滑入窗口,從左邊滑出窗口。同樣,在前向緩沖區(qū)中也是相同的滑動過程。移動的字節(jié)數(shù)與標記中編碼的字符數(shù)相等。
lz77_compress的時間復雜度為O(n),其中n是原始數(shù)據(jù)中符號的個數(shù)。這是因為,對于數(shù)據(jù)中每個n/c個編碼的標記,其中1/c是一個代表編碼效率的常量因素,調(diào)用一次compare_win。函數(shù)compare_win運行一段固定的時間,因為滑動窗口和前向緩沖區(qū)的大小均為常數(shù)。然而,這些常量比較大,會對lz77_compress的總體運行時間產(chǎn)生較大的影響。所以,lz77_compress的時間復雜度是O(n),但其實際的復雜度會受其常量因子的影響。這就解釋了為什么在用lz77進行數(shù)據(jù)壓縮時速度非常慢。
lz77_uncompress
lz77_uncompress操作解壓縮由lz77_compress壓縮的數(shù)據(jù)。首先,該函數(shù)從壓縮數(shù)據(jù)中讀取字符,并初始化滑動窗口和前向緩沖區(qū)。
解壓縮過程在一個循環(huán)中執(zhí)行,此循環(huán)會持續(xù)迭代執(zhí)行直到所有的符號處理完。使用ipos來保存向壓縮數(shù)據(jù)中寫入的當前位,并用opos來保存寫入恢復數(shù)據(jù)緩沖區(qū)中當前字節(jié)。在循環(huán)的每次迭代過程中,首先從壓縮數(shù)據(jù)讀取一位來確定要解碼的標記類型。
在解析一個標記時,如果讀取的首位是1,說明遇到了一個短語標記。此時讀取它的每個成員,查找滑動窗口中的短語,然后將短語寫入恢復數(shù)據(jù)緩沖區(qū)中。當查找每個短語時,調(diào)用網(wǎng)絡(luò)函數(shù)ntohl來保證窗口中的偏移量和長度的字節(jié)順序是與操作系統(tǒng)匹配的。這個轉(zhuǎn)換過程是必要的,因為從壓縮數(shù)據(jù)中讀取出來的偏移量和長度是大端格式的。在數(shù)據(jù)被拷貝到滑動窗口之前,前向緩沖區(qū)被用做一個臨時轉(zhuǎn)換區(qū)來保存數(shù)據(jù)。最后,寫入該標記編碼的匹配的符號。如果讀取的標記的首位是0,說明遇到了一個符號標記。在這種情況下,將該標記編碼的匹配符號寫入恢復數(shù)據(jù)緩沖區(qū)中。
一旦將解碼的數(shù)據(jù)寫入恢復數(shù)據(jù)的緩沖區(qū)中,就調(diào)整滑動窗口。要將數(shù)據(jù)通過滑動窗口,將數(shù)據(jù)從右邊滑入窗口,從左邊滑出窗口。移動的字節(jié)數(shù)與從標記中解碼的字符數(shù)相等。
lz77_uncompress的時間復雜度為O(n),其中n是原始數(shù)據(jù)中符號的個數(shù)。
-
數(shù)據(jù)
+關(guān)注
關(guān)注
8文章
6808瀏覽量
88743
原文標題:數(shù)據(jù)壓縮算法:LZ77 算法的分析與實現(xiàn)
文章出處:【微信號:TheAlgorithm,微信公眾號:算法與數(shù)據(jù)結(jié)構(gòu)】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
相關(guān)推薦
評論