android圖片壓縮總結
android圖片壓縮無非兩種,一種質量壓縮,一種像素壓縮,前者多用于圖片上傳時,后者多用于本地圖片展示縮略圖時。
對于質量壓縮,主要用到的一個方法就是:
public boolean compress(CompressFormat format, int quality, OutputStream stream) {} 這是Bitmap類里的一個方法,第一個參數表示圖片壓縮的格式,android中提供了以下格式:
public enum CompressFormat {
JPEG (0),
PNG (1),
WEBP (2);
CompressFormat(int nativeInt) {
this.nativeInt = nativeInt;
}
final int nativeInt;
} 第二個參數表示壓縮的質量,注意這個是壓縮的關鍵,它的取值是0到100,越小表示壓縮的越厲害,第三個參數表示把壓縮的數據寫入了outputstream流中。
OK,來看一個例子:
public byte [] compressBitmap(Bitmap bitmap,int max){
int quality = 100;
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
bitmap.compress(Bitmap.CompressFormat.JPEG,quality,byteArrayOutputStream);
while (byteArrayOutputStream.toByteArray().length / 1024 > max){
byteArrayOutputStream.reset();
quality = quality -10;
bitmap.compress(Bitmap.CompressFormat.JPEG,quality,byteArrayOutputStream);
}
return byteArrayOutputStream.toByteArray();
} 這個方法的第一個參數不必解釋,第二個參數表示你要求的壓縮后圖片最大可以是多少。最后可以拿到一個byte數組。我們有了這個byte數組就可以轉化為file或者bitmap。
注意這種質量壓縮后,像素本身沒有改變。
對于像素壓縮,顧名思義就是壓縮像素。這里用到的一個主要的方法就是:
public static Bitmap decodeFile(String pathName, Options opts) {
Bitmap bm = null;
InputStream stream = null;
try {
stream = new FileInputStream(pathName);
bm = decodeStream(stream, null, opts);
} catch (Exception e) {
/* do nothing.
If the exception happened on open, bm will be null.
*/
Log.e("BitmapFactory", "Unable to decode stream: " + e);
} finally {
if (stream != null) {
try {
stream.close();
} catch (IOException e) {
// do nothing here
}
}
}
return bm;
} 這是BitmapFactory中的一個靜態方法,第一個參數表示file的全路徑,第二個參數是關鍵,Options是BitmapFactory類中的一個靜態內部類,它有兩個非常重要的屬性:
/** * If set to true, the decoder will return null (no bitmap), but * the out... fields will still be set, allowing the caller to query * the bitmap without having to allocate the memory for its pixels. */ public boolean inJustDecodeBounds; /** * If set to a value > 1, requests the decoder to subsample the original * image, returning a smaller image to save memory. The sample size is * the number of pixels in either dimension that correspond to a single * pixel in the decoded bitmap. For example, inSampleSize == 4 returns * an image that is 1/4 the width/height of the original, and 1/16 the * number of pixels. Any value <= 1 is treated the same as 1. Note: the * decoder uses a final value based on powers of 2, any other value will * be rounded down to the nearest power of 2. */ public int inSampleSize;
第一個屬性inJustDecodeBounds,如果設置為ture,則返回null。
第二個屬性inSampleSize表示縮放比例,大于1表示縮小了原來的多少,比如inSampleSize == 4,就表示縮小了原來的四分之一,如果小于1則和1相同。
好了來看看這個網上遍地都是的一個像素壓縮的方法:
private Bitmap getimage(String srcPath) {
BitmapFactory.Options newOpts = new BitmapFactory.Options();
//開始讀入圖片,此時把options.inJustDecodeBounds 設回true了
newOpts.inJustDecodeBounds = true;
Bitmap bitmap = BitmapFactory.decodeFile(srcPath,newOpts);//此時返回bm為空
newOpts.inJustDecodeBounds = false;
int w = newOpts.outWidth;
int h = newOpts.outHeight;
//現在主流手機比較多是800*480分辨率,所以高和寬我們設置為
float hh = 800f;//這里設置高度為800f
float ww = 480f;//這里設置寬度為480f
//縮放比。由于是固定比例縮放,只用高或者寬其中一個數據進行計算即可
int be = 1;//be=1表示不縮放
if (w > h && w > ww) {//如果寬度大的話根據寬度固定大小縮放
be = (int) (newOpts.outWidth / ww);
} else if (w < h && h > hh) {//如果高度高的話根據寬度固定大小縮放
be = (int) (newOpts.outHeight / hh);
}
if (be <= 0)
be = 1;
newOpts.inSampleSize = be;//設置縮放比例
//重新讀入圖片,注意此時已經把options.inJustDecodeBounds 設回false了
bitmap = BitmapFactory.decodeFile(srcPath, newOpts);
return bitmap;
} 是不是就很好理解了。
本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!