C++實(shí)現(xiàn)圖像壓縮的示例代碼
一、問(wèn)題描述
一幅圖像由n*m個(gè)像素點(diǎn)組成,其中每個(gè)像素的灰度值范圍是0~255(即需要8bit來(lái)存儲(chǔ)像素的灰度值),由此可以得出存儲(chǔ)此圖像需要的存儲(chǔ)空間為n*m*8(bit)。
不難看出,直接采用上述的存儲(chǔ)方式是需要占用很多存儲(chǔ)空間的。此時(shí)可以采用圖像壓縮算法來(lái)節(jié)省存儲(chǔ)空間。
二、算法分析
1、算法思想
將像素序列分段,段內(nèi)的像素灰度值相似(可以用小于8bit的空間來(lái)存儲(chǔ)一個(gè)像素灰度值),一段內(nèi)的像素用相同的bit數(shù)來(lái)存儲(chǔ),只需要額外存儲(chǔ)每段的長(zhǎng)度和bit數(shù)即可,這樣可以節(jié)省很多空間
于是問(wèn)題的關(guān)鍵就在于如何分段,使得存儲(chǔ)空間的占用最少
2、算法實(shí)現(xiàn)
假設(shè)像素點(diǎn)的灰度集合為p[n],創(chuàng)建三個(gè)表 s[i]、l[i]、b[i],其中:
- s[i]來(lái)記錄前 i 個(gè)數(shù)字的最優(yōu)處理方式得到的最優(yōu)解。
- l[i]來(lái)記錄當(dāng)前第 i 個(gè)數(shù)所在組中有多少個(gè)數(shù)。
- b[i]中存放前 i 個(gè)像素點(diǎn)最后一段位數(shù)的最大值。
假設(shè)產(chǎn)生了m個(gè)段,則存儲(chǔ)第i段像素所需要的空間為 : l[i] * b[i] + 11(l[i] * b[i]表示這一段像素本身需要的信息, 11則表示這一段的長(zhǎng)度l[i]以及該段像素每一個(gè)都用幾位來(lái)表示b[i], 即3 + 8 = 11位)
總存儲(chǔ)空間為 11m+∑ l[i]*b[i]
此時(shí)只要找出最優(yōu)數(shù)組,即可得到最有效的壓縮方法。最優(yōu)數(shù)組含義是:s[i],1≤i≤n,是像素序列{p1,…,pi}的最優(yōu)分段所需的存儲(chǔ)位數(shù)。
遞推關(guān)系式:
三、代碼實(shí)現(xiàn)
#include<iostream> using namespace std; const int N = 10; void show(int s[], int l[], int b[], int n) { //在輸出s[n]存儲(chǔ)位數(shù)后,s[]數(shù)組則被重新賦值,用來(lái)存儲(chǔ)分段的位置 cout << "圖像壓縮后的最小空間為:" << s[n] << endl; } int length(int x) { int count = 1; x = x / 2; while (x > 0) { count++; x = x / 2; } return count; } void compress(int n,int p[N], int s[N], int l[N], int b[N]) { int lmax = 256,bmax; int header = 11; s[0] = 0; for (int i = 1; i <= n; i++) { b[i] = length(p[i]); bmax = b[i]; s[i] = s[i - 1] + bmax; l[i] = 1; for (int j = 2; j <= i && j <= lmax; j++) { if (bmax < b[i - j + 1]) { bmax = b[i - j + 1]; } if (s[i] > s[i - j] + j * bmax) { s[i] = s[i - j] + j * bmax; l[i] = j; } } s[i] += header; } } int main() { int data[N] = { 10,12,11,9,145,238,2,3,5,1 }; cout << "圖像的灰度值序列:"; for (int i = 0; i < N; i++) { cout << data[i]<<" "; } cout << endl; int s[N],l[N], b[N]; compress(N - 1, data, s, b, l); show(s,l,b,N - 1); }
運(yùn)行結(jié)果:
到此這篇關(guān)于C++實(shí)現(xiàn)圖像壓縮的示例代碼的文章就介紹到這了,更多相關(guān)C++圖像壓縮內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++使用cuBLAS加速矩陣乘法運(yùn)算的實(shí)現(xiàn)代碼
這篇文章主要介紹了C++使用cuBLAS加速矩陣乘法運(yùn)算,將cuBLAS庫(kù)的乘法運(yùn)算進(jìn)行了封裝,方便了算法調(diào)用,具體實(shí)現(xiàn)代碼跟隨小編一起看看吧2021-09-09Qt qml實(shí)現(xiàn)動(dòng)態(tài)輪播圖效果
這篇文章主要為大家詳細(xì)介紹了Qt和qml實(shí)現(xiàn)動(dòng)態(tài)輪播圖效果的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,有需要的小伙伴可以參考一下2024-12-12Qt創(chuàng)建SQlite數(shù)據(jù)庫(kù)的示例代碼
本文主要介紹了Qt創(chuàng)建SQlite數(shù)據(jù)庫(kù)的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05OpenCV實(shí)現(xiàn)特征檢測(cè)和特征匹配方法匯總
一幅圖像中總存在著其獨(dú)特的像素點(diǎn),這些點(diǎn)我們可以認(rèn)為就是這幅圖像的特征,成為特征點(diǎn),本文主要介紹了OpenCV實(shí)現(xiàn)特征檢測(cè)和特征匹配方法,感興趣的可以了解一下2021-08-08C語(yǔ)言詳細(xì)分析講解struct與union使用方法
最近開(kāi)始自學(xué)C語(yǔ)言,從最基礎(chǔ)部分的開(kāi)始學(xué)起。今天看書(shū)的時(shí)候注意到了struct和union似乎很像,除了名字不同,看起來(lái)幾乎沒(méi)有區(qū)別。<BR>既然C中定義了struct和union兩個(gè)關(guān)鍵字,那么它們肯定是有區(qū)別的,在查了一些資料之后我來(lái)總結(jié)一下他們的使用2022-04-04C++如何判斷一個(gè)數(shù)字是否為質(zhì)數(shù)
這篇文章主要為大家詳細(xì)介紹了C++如何判斷一個(gè)數(shù)字是否為質(zhì)數(shù),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-03-03Qt利用QJson實(shí)現(xiàn)解析數(shù)組的示例詳解
這篇文章主要為大家詳細(xì)介紹了Qt如何利用QJson實(shí)現(xiàn)解析數(shù)組功能,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Qt有一定幫助,需要的小伙伴可以了解一下2022-10-10利用C++11原子量如何實(shí)現(xiàn)自旋鎖詳解
當(dāng)自旋鎖嘗試獲取鎖時(shí)以忙等待(busy waiting)的形式不斷地循環(huán)檢查鎖是否可用,下面這篇文章主要給大家介紹了關(guān)于利用C++11原子量如何實(shí)現(xiàn)自旋鎖的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2018-06-06