Android三種常見的圖片壓縮方式
下面就為大家?guī)?種比較常見的壓縮方式
先給出一組數(shù)據(jù)
原圖:width:2976; height:2976
原圖實(shí)際:--->byte:2299820 Mb:2.19328
質(zhì)量壓縮 size--->:byte:1599831 kb:1562.33496
按比例壓縮 size--->:byte:191707 kb:187.21387
魯班壓縮 size--->:byte:143792 kb:140.42188
壓縮效果:魯班壓縮 > 按比例壓縮 > 質(zhì)量壓縮
1、質(zhì)量壓縮
public void getBitmap(String imgPath, String outPath) {
BitmapFactory.Options newOpts = new BitmapFactory.Options();
newOpts.inJustDecodeBounds = false;
newOpts.inPurgeable = true;
newOpts.inInputShareable = true;
// Do not compress
newOpts.inSampleSize = 1;
newOpts.inPreferredConfig = Config.RGB_565;
storeImage(bitmap, outPath); //保存圖片
}
注意
- 質(zhì)量壓縮不會(huì)減少圖片的像素,它是在保持像素的前提下改變圖片的位深及透明度等,來達(dá)到壓縮圖片的目的,這也是為什么該方法叫質(zhì)量壓縮方法。所以這種方法,很可能不會(huì)減少圖片的大小
- 如果是bit.compress(CompressFormat.PNG, quality, baos);這樣的png格式,quality就沒有作用了,bytes.length不會(huì)變化,因?yàn)閜ng圖片是無損的,不能進(jìn)行壓縮
保存圖片
/**
* 把bitmap轉(zhuǎn)化成圖片存儲(chǔ)在本地
*
* @param bitmap
* @param outPath 本地的存儲(chǔ)路徑
* @throws FileNotFoundException
*/
public static boolean storeImage(Bitmap bitmap, String outPath) throws FileNotFoundException {
FileOutputStream os = new FileOutputStream(outPath);
boolean compressResult = bitmap.compress(Bitmap.CompressFormat.JPEG, 100, os);
return compressResult;
}
2、按比例壓縮(尺寸壓縮、采樣率壓縮)
/**
* 按比例壓縮
*
* @param path 原圖片路徑
* @param targetW 壓縮后寬度
* @param targetH 壓縮后高度
* @return 壓縮后的圖片的保存路徑
*/
public static String compressScale(String path,, String outPath, int targetW, int targetH) throws FileNotFoundException {
// 獲取option
BitmapFactory.Options options = new BitmapFactory.Options();
// inJustDecodeBounds設(shè)置為true,這樣使用該option decode出來的Bitmap是null,
// 只是把長寬存放到option中
options.inJustDecodeBounds = true;
// 此時(shí)bitmap為null
Bitmap bitmap = BitmapFactory.decodeFile(path, options);
int inSampleSize = 1; // 1是不縮放
// 計(jì)算寬高縮放比例
int inSampleSizeW = options.outWidth / targetW;
int inSampleSizeH = options.outHeight / targetH;
// 最終取大的那個(gè)為縮放比例,這樣才能適配,例如寬縮放3倍才能適配屏幕,而
// 高不縮放就可以,那樣的話如果按高縮放,寬在屏幕內(nèi)就顯示不下了
if (inSampleSizeW > inSampleSizeH) {
inSampleSize = inSampleSizeW;
} else {
inSampleSize = inSampleSizeH;
}
// 一定要記得將inJustDecodeBounds設(shè)為false,否則Bitmap為null
options.inJustDecodeBounds = false;
// 設(shè)置縮放比例(采樣率)
options.inSampleSize = inSampleSize;
bitmap = BitmapFactory.decodeFile(path, options);
boolean isSuccess = storeImage(bitmap, outPath);
if (isSuccess) {
return outPath;
}
return "";
}
這種方法是設(shè)置圖片的采樣率,降低圖片像素,通過縮放圖片像素來減少圖片大小。
那么我又是如何來獲取圖片壓縮前后的大小的呢?
注意:這里的圖片的大小是指圖片的實(shí)際大小,不是bitmap在內(nèi)存中所占的大小,看壓縮效果,就得看圖片在文件中所占的大小。
/**
* 獲取本地文件大小
*
* @param imgPath 圖片的路徑
* @return 圖片實(shí)際的大小,單位byte
*/
public static int getFileSize(String imgPath) {
int size = 0;
try {
FileInputStream fis = new FileInputStream(new File(imgPath));
size = fis.available();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return size;
}
3、魯班壓縮(推薦)
魯班壓縮是根據(jù)一種算法得出來的,壓縮效果基本上和微信一致,差不多在200k以內(nèi),圖片不失真。
魯班壓縮:
https://github.com/Curzibn/Luban
build.gradle添加依賴
compile 'top.zibin:Luban:1.1.3'
private void lunBanPress(String path) {
String pressPath = Environment.getExternalStorageDirectory().getPath();
Luban.with(this)
.load(path) // 傳人要壓縮的圖片列表
.ignoreBy(100) // 忽略不壓縮圖片的大小
.setTargetDir(pressPath) // 設(shè)置壓縮后文件存儲(chǔ)位置
.setCompressListener(new OnCompressListener() { //設(shè)置回調(diào)
@Override
public void onStart() {
// TODO 壓縮開始前調(diào)用,可以在方法內(nèi)啟動(dòng) loading UI
Log.i(TAG, "onStart:開始魯班壓縮 ");
}
@Override
public void onSuccess(File file) {
// TODO 壓縮成功后調(diào)用,返回壓縮后的圖片文件
Glide.with(activity).load(file).into(iv2);
Log.i(TAG, "onSuccess: 魯班壓縮成功 :");
try {
int size = new FileInputStream(file).available();
Log.i("tag", "魯班壓縮 size--->:" + "byte:" + size + " kb:"
+ (float) size / 1024);
} catch (IOException e) {
e.printStackTrace();
}
}
@Override
public void onError(Throwable e) {
// TODO 當(dāng)壓縮過程出現(xiàn)問題時(shí)調(diào)用
Log.i(TAG, "onError: 魯班壓縮出錯(cuò)");
}
}).launch(); //啟動(dòng)壓縮
}
源碼地址:
https://github.com/zhouxu88/ImgCompress
到這里就結(jié)束啦~
以上就是Android三種常見的圖片壓縮方式的詳細(xì)內(nèi)容,更多關(guān)于Android 圖片壓縮的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Android源碼學(xué)習(xí)之單例模式應(yīng)用及優(yōu)點(diǎn)介紹
動(dòng)態(tài)確保某一個(gè)類只有一個(gè)實(shí)例,而且自行實(shí)例化并向整個(gè)系統(tǒng)提供這個(gè)實(shí)例這就是Android單例模式應(yīng)用,接下來詳細(xì)介紹,有需求的朋友可以參考下2013-01-01
Android獲取應(yīng)用程序名稱(ApplicationName)示例
本文以實(shí)例方式為大家介紹下獲取應(yīng)用程序名稱(ApplicationName)的具體實(shí)現(xiàn),感興趣的各位可以參考下哈2013-06-06
Android通過自定義Activity實(shí)現(xiàn)懸浮的Dialog詳解
flutter日期選擇器 flutter時(shí)間選擇器
淺析Android手機(jī)衛(wèi)士保存手機(jī)安全號(hào)碼
Android ImageView實(shí)現(xiàn)圖片裁剪和顯示功能
Android開發(fā)疫情查詢app(實(shí)例代碼)
Android音頻開發(fā)之SurfaceView的使用詳解

