一、前言:
成都創(chuàng)新互聯(lián)成立與2013年,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項(xiàng)目網(wǎng)站制作、成都做網(wǎng)站網(wǎng)站策劃,項(xiàng)目實(shí)施與項(xiàng)目整合能力。我們以讓每一個(gè)夢(mèng)想脫穎而出為使命,1280元輝南做網(wǎng)站,已為上家服務(wù),為輝南各地企業(yè)和個(gè)人服務(wù),聯(lián)系電話:13518219792
2.質(zhì)量壓縮
注意:
第二次壓縮之前都要先清空 baos.reset(); 再進(jìn)行壓縮 image.compress(Bitmap.CompressFormat.JPEG, quality, baos);
有時(shí)候我們采用質(zhì)量壓縮沒有效果,有可能是每次壓縮的質(zhì)量過小,所以我們可以嘗試修改壓縮質(zhì)量(quality)是10;
quality壓縮機(jī)提示,0-100。0表示壓縮
小尺寸,100意味著最大質(zhì)量的壓縮。一些
格式,如無損的PNG,將忽略質(zhì)量設(shè)定;
3.混合方式壓縮
鏈接:
1:ByteArrayOutputStream 壓縮圖片質(zhì)量,不會(huì)改變圖片大小
private?Bitmap?compressImage(Bitmap?image)?{
ByteArrayOutputStream?baos?=?new?ByteArrayOutputStream();
image.compress(Bitmap.CompressFormat.JPEG,?100,?baos);//質(zhì)量壓縮方法,這里100表示不壓縮,把壓縮后的數(shù)據(jù)存放到baos中
int?options?=?100;????????while?(?baos.toByteArray().length?/?1024100)?{?//循環(huán)判斷如果壓縮后圖片是否大于100kb,大于繼續(xù)壓縮
baos.reset();//重置baos即清空baos
image.compress(Bitmap.CompressFormat.JPEG,?options,?baos);//這里壓縮options%,把壓縮后的數(shù)據(jù)存放到baos中
options?-=?10;//每次都減少10????????}
ByteArrayInputStream?isBm?=?new?ByteArrayInputStream(baos.toByteArray());//把壓縮后的數(shù)據(jù)baos存放到ByteArrayInputStream中
Bitmap?bitmap?=?BitmapFactory.decodeStream(isBm,?null,?null);//把ByteArrayInputStream數(shù)據(jù)生成圖片
return?bitmap;
}
2:就是改變圖片大小
private?Bitmap?getimage(String?srcPath)?{
BitmapFactory.Options?newOpts?=?new?BitmapFactory.Options();????????//開始讀入圖片,此時(shí)把options.inJustDecodeBounds?設(shè)回true了
newOpts.inJustDecodeBounds?=?true;
Bitmap?bitmap?=?BitmapFactory.decodeFile(srcPath,newOpts);//此時(shí)返回bm為空
newOpts.inJustDecodeBounds?=?false;????????int?w?=?newOpts.outWidth;????????int?h?=?newOpts.outHeight;????????//現(xiàn)在主流手機(jī)比較多是800*480分辨率,所以高和寬我們?cè)O(shè)置為
float?hh?=?800f;//這里設(shè)置高度為800f
float?ww?=?480f;//這里設(shè)置寬度為480f????????//縮放比。由于是固定比例縮放,只用高或者寬其中一個(gè)數(shù)據(jù)進(jìn)行計(jì)算即可
int?be?=?1;//be=1表示不縮放
if?(w??h??w??ww)?{//如果寬度大的話根據(jù)寬度固定大小縮放
be?=?(int)?(newOpts.outWidth?/?ww);
}?else?if?(w??h??h??hh)?{//如果高度高的話根據(jù)寬度固定大小縮放
be?=?(int)?(newOpts.outHeight?/?hh);
}????????if?(be?=?0)
be?=?1;
newOpts.inSampleSize?=?be;//設(shè)置縮放比例????????//重新讀入圖片,注意此時(shí)已經(jīng)把options.inJustDecodeBounds?設(shè)回false了
bitmap?=?BitmapFactory.decodeFile(srcPath,?newOpts);????????return?compressImage(bitmap);//壓縮好比例大小后再進(jìn)行質(zhì)量壓縮????}
老規(guī)矩 先上依賴
Add it in your root build.gradle at the end of repositories:
allprojects {
repositories {
... ??
maven { url '' }
}
}
dependencies {
implementation 'com.github.asnhkl1:X-CompressImage:1.0.1'
}
背景:
對(duì)于一些需要對(duì)圖片進(jìn)行定制化壓縮的場(chǎng)景,如必須圖片必須多少多少K以下,或者分辨率有要求的場(chǎng)景。開發(fā)此庫(kù)。
你可以這樣用:
CompressConfig ?為建造模式去設(shè)置參數(shù)
CompressConfig compressConfig = CompressConfig.builder()
? ? ? ? ? ? .setUnCompressMinPixel(1000) // 最小像素不壓縮,默認(rèn)值:1000
? ? ? ? ? .setUnCompressNormalPixel(2000) // 標(biāo)準(zhǔn)像素不壓縮,默認(rèn)值:2000
? ? ? ? ? .setMaxPixel(1200) // 長(zhǎng)或?qū)挷怀^的最大像素 (單位px),默認(rèn)值:1200
? ? ? ? ? .setMaxSize(200 * 1024) // 壓縮到的最大大小 (單位B),默認(rèn)值:200 * 1024 = 200KB
? ? ? ? ? .enablePixelCompress(true) // 是否啟用像素壓縮,默認(rèn)值:true
? ? ? ? ? .enableQualityCompress(true) // 是否啟用質(zhì)量壓縮,默認(rèn)值:true
? ? ? ? ? .enableReserveRaw(false) // 是否保留源文件,默認(rèn)值:true
? ? ? ? ? .setCacheDir(path) // 壓縮后緩存圖片路徑,默認(rèn)值:Constants.COMPRESS_CACHE
? ? ? ? ? .setShowCompressDialog(true) // 是否顯示壓縮進(jìn)度條,默認(rèn)值:false
? ? ? ? ? .create();
如果你僅僅只是壓縮圖片 ?我們給有默認(rèn)的參數(shù)使用getDefaultConfig()
this.unCompressMinPixel =1000;
this.unCompressNormalPixel =2000;
this.maxPixel =1200;
this.maxSize =204800;
this.enablePixelCompress =true;
this.enableQualityCompress =true;
this.enableReserveRaw =true;
下邊就是使用了 。很簡(jiǎn)單 ,傳參加回調(diào)
CompressImageManager.build(this,compressConfig, photos, new CompressImage.CompressListener() {
? ? ? @Override
? ? ? public void onCompressSuccess(ArrayListPhoto var1) {
? ? ? ? ? Log.i("imageCompress","success");
? ? ? ? //do what you want to do
? ? ? }
? ? ? @Override
? ? ? public void onCompressFailed(ArrayListPhoto var1, String var2) {
? ? ? ? ? Log.e("imageCompress","false",null);
? ? ? }
? }).compress();
git 地址:
歡迎指教,互相學(xué)習(xí)
圖片壓縮就是為了避免我們內(nèi)存的溢出。而BitMap是android系統(tǒng)中對(duì)圖像處理最重要的一個(gè)類,所以我們可以用他來對(duì)圖像進(jìn)行剪切,壓縮,一系列操作。常用的壓縮方法有質(zhì)量壓縮,采樣率壓縮,縮放法壓縮,RGB_565壓縮等。
質(zhì)量壓縮
質(zhì)量壓縮不會(huì)減少圖片的像素,他是在保持像素的前提下來改變圖片的透明度,但是圖片的長(zhǎng),寬,像素都不會(huì)變,所以他占用的內(nèi)存也不會(huì)改變。
采樣率壓縮
采樣率壓縮其實(shí)就是縮放bitamp的尺寸,通過調(diào)節(jié)其inSampleSize參數(shù),比如調(diào)節(jié)為2,寬高會(huì)為原來的1/2,寬高都減少了,內(nèi)存自然也就減少了。
縮放法壓縮
放縮法壓縮使用的是通過矩陣對(duì)圖片進(jìn)行裁剪,也是通過縮放圖片尺寸,來達(dá)到壓縮圖片的效果,和采樣率的原理一樣
RGB_565壓縮
這是通過壓縮像素占用的內(nèi)存來達(dá)到壓縮的效果,一般不建議使用ARGB_4444,因?yàn)楫嬞|(zhì)實(shí)在是辣雞,如果對(duì)透明度沒有要求,建議可以改成RGB_565,相比ARGB_8888將節(jié)省一半的內(nèi)存開銷。
圖片二次采樣壓縮
1.第一次采樣
inJustDecodeBounds設(shè)置為ture,只會(huì)加載圖片的邊框,不會(huì)加載圖片的具體內(nèi)容。
2.根據(jù)原圖的寬高,結(jié)合控件的寬高計(jì)算出縮放比例,然后通過inSamplesize(in 三pou Size)對(duì)圖片進(jìn)行尺寸的縮放。
3.二次采樣時(shí)需要將圖片加載出來顯示,不能只加載圖片的框架,因此inJustDecodeBounds屬性要設(shè)置為false。
一、支持自定義配置、不失真和批量處理
二、圖片上傳為什么要壓縮
1、圖片服務(wù)器空間限制,磁盤昂貴
2、網(wǎng)絡(luò)不穩(wěn)定,大文件需要斷點(diǎn)續(xù)傳
3、盡可能避免安卓OOM異常
4、后臺(tái)約定的規(guī)則200KB
5、需要上傳原圖的應(yīng)用有醫(yī)院臨床項(xiàng)目、金融銀行
三、圖片壓縮流程
1、遞歸每張圖片
2、設(shè)置圖片格式 Bitmap.CompressFormat.JPG
png, jpg,webp
3、質(zhì)量壓縮bitmap.compress(format,quality,baos)
由于png是無損壓縮,所以設(shè)置quality無效(不適合作為縮略圖)
采樣率壓縮BitmapFactory.Options.inSampleSize
縮小圖片分辨率,減少所占用磁盤空間和內(nèi)存大小
縮放壓縮canvas.drawBitmap(bitmap, null,rectF,null)
減少圖片的像素,降低所占用磁盤空間大小和內(nèi)存大小,可用于緩存縮略圖
JNI調(diào)用JPEG庫(kù)
Android的圖片引擎使用的是閹割版的skia引擎,去掉了圖片壓縮中的哈夫曼算法
4、像素修復(fù)
5、返回壓縮
6、完成壓縮
demo:
參考:
Luban框架
缺點(diǎn)
1、當(dāng)沒有設(shè)定壓縮路徑時(shí),拋異常無閃退
2、源碼中,壓縮比率固定值60,無法修改
3、壓縮配置,參數(shù)不太適應(yīng)真實(shí)項(xiàng)目需求
4、不能指定壓縮大小,比如100KB以內(nèi)
簡(jiǎn)單吹下牛:很多app都會(huì)要加載圖片,但是如果不壓縮圖片就很容易OOM,
個(gè)人看來OOM 出現(xiàn)原因總的來說分為兩種:
一種是內(nèi)存溢出(好像在扯淡,OOM本身就是內(nèi)存溢出)
另一種是:圖片過大,一個(gè)屏幕顯示不完全造成,似乎也是一。。 如有錯(cuò)誤純屬扯淡;
為了避免上面的情況:加載圖片的時(shí)候可以進(jìn)行壓縮,上傳的時(shí)候要可以進(jìn)行壓縮,在圖片不可見的時(shí)候進(jìn)行回收(onDetach()),再吹一句 用了fresco+壓縮之后加載圖片完全沒問題了。
一、質(zhì)量壓縮方法:
privateBitmap compressImage(Bitmap image) {
ByteArrayOutputStream?baos?=newByteArrayOutputStream();
image.compress(Bitmap.CompressFormat.JPEG,100,?baos);//質(zhì)量壓縮方法,這里100表示不壓縮,把壓縮后的數(shù)據(jù)存放到baos中
intoptions?=100;
while(?baos.toByteArray().length?/1024100)?{//循環(huán)判斷如果壓縮后圖片是否大于100kb,大于繼續(xù)壓縮
baos.reset();//重置baos即清空baos
image.compress(Bitmap.CompressFormat.JPEG,?options,?baos);//這里壓縮options%,把壓縮后的數(shù)據(jù)存放到baos中
options?-=10;//每次都減少10
}
ByteArrayInputStream?isBm?=newByteArrayInputStream(baos.toByteArray());//把壓縮后的數(shù)據(jù)baos存放到ByteArrayInputStream中
Bitmap?bitmap?=?BitmapFactory.decodeStream(isBm,null,null);//把ByteArrayInputStream數(shù)據(jù)生成圖片
returnbitmap;
}
二、圖片按比例大小壓縮方法(根據(jù)Bitmap圖片壓縮)
privateBitmap comp(Bitmap image) {
ByteArrayOutputStream?baos?=newByteArrayOutputStream();
image.compress(Bitmap.CompressFormat.JPEG,100,?baos);
if(?baos.toByteArray().length?/10241024)?{//判斷如果圖片大于1M,進(jìn)行壓縮避免在生成圖片(BitmapFactory.decodeStream)時(shí)溢出
baos.reset();//重置baos即清空baos
image.compress(Bitmap.CompressFormat.JPEG,50,?baos);//這里壓縮50%,把壓縮后的數(shù)據(jù)存放到baos中
}
ByteArrayInputStream?isBm?=newByteArrayInputStream(baos.toByteArray());
BitmapFactory.Options?newOpts?=newBitmapFactory.Options();
//開始讀入圖片,此時(shí)把options.inJustDecodeBounds?設(shè)回true了
newOpts.inJustDecodeBounds?=true;
Bitmap?bitmap?=?BitmapFactory.decodeStream(isBm,null,?newOpts);
newOpts.inJustDecodeBounds?=false;
intw?=?newOpts.outWidth;
inth?=?newOpts.outHeight;
//現(xiàn)在主流手機(jī)比較多是800*480分辨率,所以高和寬我們?cè)O(shè)置為
floathh?=?800f;//這里設(shè)置高度為800f
floatww?=?480f;//這里設(shè)置寬度為480f
//縮放比。由于是固定比例縮放,只用高或者寬其中一個(gè)數(shù)據(jù)進(jìn)行計(jì)算即可
intbe?=1;//be=1表示不縮放
if(w??h??w??ww)?{//如果寬度大的話根據(jù)寬度固定大小縮放
be?=?(int)?(newOpts.outWidth?/?ww);
}elseif(w??h??h??hh)?{//如果高度高的話根據(jù)寬度固定大小縮放
be?=?(int)?(newOpts.outHeight?/?hh);
}
if(be?=0)
be?=1;
newOpts.inSampleSize?=?be;//設(shè)置縮放比例
//重新讀入圖片,注意此時(shí)已經(jīng)把options.inJustDecodeBounds?設(shè)回false了
isBm?=newByteArrayInputStream(baos.toByteArray());
bitmap?=?BitmapFactory.decodeStream(isBm,null,?newOpts);
returncompressImage(bitmap);//壓縮好比例大小后再進(jìn)行質(zhì)量壓縮
}