Android中圖片壓縮方案詳解

圖片的展示可以說在我們?nèi)魏我粋€應(yīng)用中都避免不了缰泡,可是大量的圖片就會出現(xiàn)很多的問題娇哆,比如加載大圖片或者多圖時(shí)的OOM問題劫狠,可以移步到Android高效加載大圖夸研、多圖避免程序OOM.還有一個問題就是圖片的上傳下載問題狰域,往往我們都喜歡圖片既清楚又占的內(nèi)存小媳拴,也就是盡可能少的耗費(fèi)我們的流量黄橘,這就是我今天所要講述的問題:圖片的壓縮方案的詳解。

1屈溉、質(zhì)量壓縮法

設(shè)置bitmap options屬性塞关,降低圖片的質(zhì)量,像素不會減少
第一個參數(shù)為需要壓縮的bitmap圖片對象子巾,第二個參數(shù)為壓縮后圖片保存的位置
設(shè)置options 屬性0-100帆赢,來實(shí)現(xiàn)壓縮。

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 / 1024>100) {  //循環(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;  
    }  

質(zhì)量壓縮不會減少圖片的像素缠导。它是在保持像素不變的前提下改變圖片的位深及透明度等廉羔,來達(dá)到壓縮圖片的目的溉痢。進(jìn)過它壓縮的圖片文件大小會有改變僻造,但是導(dǎo)入成bitmap后占得內(nèi)存是不變的。因?yàn)橐3窒袼夭蛔兒⒈运蜔o法無限壓縮髓削,到達(dá)一個值之后就不會繼續(xù)變小了。顯然這個方法并不適用于縮略圖镀娶,其實(shí)也不適用于想通過壓縮圖片減少內(nèi)存的適用立膛,僅僅適用于想在保證圖片質(zhì)量的同時(shí)減少文件大小的情況而已。

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ī)比較多是1280*720分辨率,所以高和寬我們設(shè)置為  
        float hh = 1280f;//這里設(shè)置高度為1280f  
        float ww = 720f;//這里設(shè)置寬度為720f 
        //縮放比轩娶。由于是固定比例縮放儿奶,只用高或者寬其中一個數(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ì)量壓縮  
    }  

這個方法的好處是大大的縮小了內(nèi)存的使用鳄抒,在讀存儲器上的圖片時(shí)闯捎,如果不需要高清的效果,可以先只讀取圖片的邊许溅,通過寬和高設(shè)定好取樣率后再加載圖片瓤鼻,這樣就不會過多的占用內(nèi)存。

3贤重、縮放法

通過縮放圖片像素來減少圖片占用內(nèi)存大小茬祷。

  • 方式一
public static void compressBitmapToFile(Bitmap bmp, File file){
    // 尺寸壓縮倍數(shù),值越大,圖片尺寸越小
    int ratio = 2;
    // 壓縮Bitmap到對應(yīng)尺寸
    Bitmap result = Bitmap.createBitmap(bmp.getWidth() / ratio, bmp.getHeight() / ratio, Config.ARGB_8888);
    Canvas canvas = new Canvas(result);
    Rect rect = new Rect(0, 0, bmp.getWidth() / ratio, bmp.getHeight() / ratio);
    canvas.drawBitmap(bmp, null, rect, null);

    ByteArrayOutputStream baos = new ByteArrayOutputStream();
    // 把壓縮后的數(shù)據(jù)存放到baos中
    result.compress(Bitmap.CompressFormat.JPEG, 100 ,baos);
    try {  
        FileOutputStream fos = new FileOutputStream(file);  
        fos.write(baos.toByteArray());  
        fos.flush();  
        fos.close();  
    } catch (Exception e) {  
        e.printStackTrace();  
    } 
}
  • 方式二
ByteArrayOutputStream out = new ByteArrayOutputStream();  
image.compress(Bitmap.CompressFormat.JPEG, 85, out);  
float zoom = (float)Math.sqrt(size * 1024 / (float)out.toByteArray().length);  
  
 Matrix matrix = new Matrix();  
matrix.setScale(zoom, zoom);  
  
 Bitmap result = Bitmap.createBitmap(image, 0, 0, image.getWidth(), image.getHeight(), matrix, true);  
  
 out.reset();  
result.compress(Bitmap.CompressFormat.JPEG, 85, out);  
while(out.toByteArray().length > size * 1024){  
    System.out.println(out.toByteArray().length);  
    matrix.setScale(0.9f, 0.9f);  
    result = Bitmap.createBitmap(result, 0, 0, result.getWidth(), result.getHeight(), matrix, true);  
    out.reset();  
    result.compress(Bitmap.CompressFormat.JPEG, 85, out);  
}

縮放法其實(shí)很簡單并蝗,設(shè)定好matrix祭犯,在createBitmap就可以了耐朴。但是我們并不知道縮放比例,而是要求了圖片的最終大小盹憎。直接用大小的比例來做的話肯定是有問題的筛峭,用大小比例的開方來做會比較接近,但是還是有差距陪每。但是只要再做一下微調(diào)應(yīng)該就可以了影晓,微調(diào)的話就是修改過的圖片大小比最終大小還大的話,就進(jìn)行0.8的壓縮再比較檩禾,循環(huán)直到大小合適挂签。這樣就能得到合適大小的圖片,而且也能比較保證質(zhì)量盼产。

4饵婆、JNI調(diào)用libjpeg庫壓縮

JNI靜態(tài)調(diào)用 bitherlibjni.c 中的方法來實(shí)現(xiàn)壓縮Java_net_bither_util_NativeUtil_compressBitmap
net_bither_util為包名,NativeUtil為類名戏售,compressBitmap為native方法名侨核,我們只需要調(diào)用saveBitmap()方法就可以,bmp 需要壓縮的Bitmap對象, quality壓縮質(zhì)量0-100, fileName 壓縮后要保存的文件地址, optimize 是否采用哈弗曼表數(shù)據(jù)計(jì)算 品質(zhì)相差5-10倍灌灾。

jstring Java_net_bither_util_NativeUtil_compressBitmap(JNIEnv* env,
        jobject thiz, jobject bitmapcolor, int w, int h, int quality,
        jbyteArray fileNameStr, jboolean optimize) {

    AndroidBitmapInfo infocolor;
    BYTE* pixelscolor;
    int ret;
    BYTE * data;
    BYTE *tmpdata;
    char * fileName = jstrinTostring(env, fileNameStr);
    if ((ret = AndroidBitmap_getInfo(env, bitmapcolor, &infocolor)) < 0) {
        LOGE("AndroidBitmap_getInfo() failed ! error=%d", ret);
        return (*env)->NewStringUTF(env, "0");;
    }
    if ((ret = AndroidBitmap_lockPixels(env, bitmapcolor, &pixelscolor)) < 0) {
        LOGE("AndroidBitmap_lockPixels() failed ! error=%d", ret);
    }

    BYTE r, g, b;
    data = NULL;
    data = malloc(w * h * 3);
    tmpdata = data;
    int j = 0, i = 0;
    int color;
    for (i = 0; i < h; i++) {
        for (j = 0; j < w; j++) {
            color = *((int *) pixelscolor);
            r = ((color & 0x00FF0000) >> 16);
            g = ((color & 0x0000FF00) >> 8);
            b = color & 0x000000FF;
            *data = b;
            *(data + 1) = g;
            *(data + 2) = r;
            data = data + 3;
            pixelscolor += 4;

        }

    }
    AndroidBitmap_unlockPixels(env, bitmapcolor);
    int resultCode= generateJPEG(tmpdata, w, h, quality, fileName, optimize);
    free(tmpdata);
    if(resultCode==0){
        jstring result=(*env)->NewStringUTF(env, error);
        error=NULL;
        return result;
    }
    return (*env)->NewStringUTF(env, "1"); //success
}

5搓译、質(zhì)量壓縮+采樣率壓縮+JNI調(diào)用libjpeg庫壓縮結(jié)合使用

首先通過尺寸壓縮,壓縮到手機(jī)常用的一個分辨率(1280*960 微信好像是壓縮到這個分辨率),然后我們要把圖片壓縮到一定大小以內(nèi)(比如說200k)锋喜,然后通過循環(huán)進(jìn)行質(zhì)量壓縮來計(jì)算options需要設(shè)置為多少些己,最后調(diào)用JNI壓縮。效果如下圖:


1.png
  • 計(jì)算縮放比
   /**
     * 計(jì)算縮放比
     * @param bitWidth 當(dāng)前圖片寬度
     * @param bitHeight 當(dāng)前圖片高度
     * @return int 縮放比
     */
    public static int getRatioSize(int bitWidth, int bitHeight) {
        // 圖片最大分辨率
        int imageHeight = 1280;
        int imageWidth = 960;
        // 縮放比
        int ratio = 1;
        // 縮放比,由于是固定比例縮放嘿般,只用高或者寬其中一個數(shù)據(jù)進(jìn)行計(jì)算即可
        if (bitWidth > bitHeight && bitWidth > imageWidth) {
            // 如果圖片寬度比高度大,以寬度為基準(zhǔn)
            ratio = bitWidth / imageWidth;
        } else if (bitWidth < bitHeight && bitHeight > imageHeight) {
            // 如果圖片高度比寬度大段标,以高度為基準(zhǔn)
            ratio = bitHeight / imageHeight;
        }
        // 最小比率為1
        if (ratio <= 0)
            ratio = 1;
        return ratio;
    }
  • 質(zhì)量壓縮+JNI壓縮
/**
     * @Description: 通過JNI圖片壓縮把Bitmap保存到指定目錄
     * @param curFilePath
     *            當(dāng)前圖片文件地址
     * @param targetFilePath
     *            要保存的圖片文件地址
     */
    public static void compressBitmap(String curFilePath, String targetFilePath) {
        // 最大圖片大小 500KB
        int maxSize = 500;
        //根據(jù)地址獲取bitmap
        Bitmap result = getBitmapFromFile(curFilePath);
        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        // 質(zhì)量壓縮方法,這里100表示不壓縮炉奴,把壓縮后的數(shù)據(jù)存放到baos中
        int quality = 100;
        result.compress(Bitmap.CompressFormat.JPEG, quality, baos);
        // 循環(huán)判斷如果壓縮后圖片是否大于500kb,大于繼續(xù)壓縮
        while (baos.toByteArray().length / 1024 > maxSize) {
            // 重置baos即清空baos
            baos.reset();
            // 每次都減少10
            quality -= 10;
            // 這里壓縮quality逼庞,把壓縮后的數(shù)據(jù)存放到baos中
            result.compress(Bitmap.CompressFormat.JPEG, quality, baos);
        }
        // JNI保存圖片到SD卡 這個關(guān)鍵
        NativeUtil.saveBitmap(result, quality, targetFilePath, true);
        // 釋放Bitmap
        if (!result.isRecycled()) {
            result.recycle();
        }
    }
  • JNI圖片壓縮工具類
package net.bither.util;
import android.graphics.Bitmap;
import android.graphics.Bitmap.Config;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Matrix;
import android.graphics.Rect;
import android.media.ExifInterface;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
/**
 * JNI圖片壓縮工具類
 *
 * @Description TODO
 * @Package net.bither.util
 * @Class NativeUtil
 */
public class NativeUtil {

    private static int DEFAULT_QUALITY = 95;

    /**
     * @Description: JNI基本壓縮
     * @param bit
     *            bitmap對象
     * @param fileName
     *            指定保存目錄名
     * @param optimize
     *            是否采用哈弗曼表數(shù)據(jù)計(jì)算 品質(zhì)相差5-10倍
     */
    public static void compressBitmap(Bitmap bit, String fileName, boolean optimize) {
        saveBitmap(bit, DEFAULT_QUALITY, fileName, optimize);
    }

    /**
     * @Description: 通過JNI圖片壓縮把Bitmap保存到指定目錄
     * @param image
     *            bitmap對象
     * @param filePath
     *            要保存的指定目錄
     */
    public static void compressBitmap(Bitmap image, String filePath) {
        // 最大圖片大小 150KB
        int maxSize = 150;
        // 獲取尺寸壓縮倍數(shù)
        int ratio = NativeUtil.getRatioSize(image.getWidth(),image.getHeight());
        // 壓縮Bitmap到對應(yīng)尺寸
        Bitmap result = Bitmap.createBitmap(image.getWidth() / ratio,image.getHeight() / ratio, Config.ARGB_8888);
        Canvas canvas = new Canvas(result);
        Rect rect = new Rect(0, 0, image.getWidth() / ratio, image.getHeight() / ratio);
        canvas.drawBitmap(image,null,rect,null);
        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        // 質(zhì)量壓縮方法,這里100表示不壓縮盆佣,把壓縮后的數(shù)據(jù)存放到baos中
        int options = 100;
        result.compress(Bitmap.CompressFormat.JPEG, options, baos);
        // 循環(huán)判斷如果壓縮后圖片是否大于100kb,大于繼續(xù)壓縮
        while (baos.toByteArray().length / 1024 > maxSize) {
            // 重置baos即清空baos
            baos.reset();
            // 每次都減少10
            options -= 10;
            // 這里壓縮options%往堡,把壓縮后的數(shù)據(jù)存放到baos中
            result.compress(Bitmap.CompressFormat.JPEG, options, baos);
        }
        // JNI保存圖片到SD卡 這個關(guān)鍵
        NativeUtil.saveBitmap(result, options, filePath, true);
        // 釋放Bitmap
        if (!result.isRecycled()) {
            result.recycle();
        }
    }

    /**
     * @Description: 通過JNI圖片壓縮把Bitmap保存到指定目錄
     * @param curFilePath
     *            當(dāng)前圖片文件地址
     * @param targetFilePath
     *            要保存的圖片文件地址
     */
    public static void compressBitmap(String curFilePath, String targetFilePath) {
        // 最大圖片大小 500KB
        int maxSize = 500;
        //根據(jù)地址獲取bitmap
        Bitmap result = getBitmapFromFile(curFilePath);
        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        // 質(zhì)量壓縮方法,這里100表示不壓縮共耍,把壓縮后的數(shù)據(jù)存放到baos中
        int quality = 100;
        result.compress(Bitmap.CompressFormat.JPEG, quality, baos);
        // 循環(huán)判斷如果壓縮后圖片是否大于500kb,大于繼續(xù)壓縮
        while (baos.toByteArray().length / 1024 > maxSize) {
            // 重置baos即清空baos
            baos.reset();
            // 每次都減少10
            quality -= 10;
            // 這里壓縮quality虑灰,把壓縮后的數(shù)據(jù)存放到baos中
            result.compress(Bitmap.CompressFormat.JPEG, quality, baos);
        }
        // JNI保存圖片到SD卡 這個關(guān)鍵
        NativeUtil.saveBitmap(result, quality, targetFilePath, true);
        // 釋放Bitmap
        if (!result.isRecycled()) {
            result.recycle();
        }

    }

    /**
     * 計(jì)算縮放比
     * @param bitWidth 當(dāng)前圖片寬度
     * @param bitHeight 當(dāng)前圖片高度
     * @return int 縮放比
     */
    public static int getRatioSize(int bitWidth, int bitHeight) {
        // 圖片最大分辨率
        int imageHeight = 1280;
        int imageWidth = 960;
        // 縮放比
        int ratio = 1;
        // 縮放比,由于是固定比例縮放,只用高或者寬其中一個數(shù)據(jù)進(jìn)行計(jì)算即可
        if (bitWidth > bitHeight && bitWidth > imageWidth) {
            // 如果圖片寬度比高度大,以寬度為基準(zhǔn)
            ratio = bitWidth / imageWidth;
        } else if (bitWidth < bitHeight && bitHeight > imageHeight) {
            // 如果圖片高度比寬度大痹兜,以高度為基準(zhǔn)
            ratio = bitHeight / imageHeight;
        }
        // 最小比率為1
        if (ratio <= 0)
            ratio = 1;
        return ratio;
    }

    /**
     * 通過文件路徑讀獲取Bitmap防止OOM以及解決圖片旋轉(zhuǎn)問題
     * @param filePath
     * @return
     */
    public static Bitmap getBitmapFromFile(String filePath){
        BitmapFactory.Options newOpts = new BitmapFactory.Options();
        newOpts.inJustDecodeBounds = true;//只讀邊,不讀內(nèi)容  
        BitmapFactory.decodeFile(filePath, newOpts);
        int w = newOpts.outWidth;
        int h = newOpts.outHeight;
        // 獲取尺寸壓縮倍數(shù)
        newOpts.inSampleSize = NativeUtil.getRatioSize(w,h);
        newOpts.inJustDecodeBounds = false;//讀取所有內(nèi)容
        newOpts.inDither = false;
        newOpts.inPurgeable=true;
        newOpts.inInputShareable=true;
        newOpts.inTempStorage = new byte[32 * 1024];
        Bitmap bitmap = null;
        File file = new File(filePath);
        FileInputStream fs = null;
        try {
            fs = new FileInputStream(file);
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }
        try {
            if(fs!=null){
                bitmap = BitmapFactory.decodeFileDescriptor(fs.getFD(),null,newOpts);
                //旋轉(zhuǎn)圖片
                int photoDegree = readPictureDegree(filePath);
                if(photoDegree != 0){
                    Matrix matrix = new Matrix();
                    matrix.postRotate(photoDegree);
                    // 創(chuàng)建新的圖片
                    bitmap = Bitmap.createBitmap(bitmap, 0, 0,
                            bitmap.getWidth(), bitmap.getHeight(), matrix, true);
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally{
            if(fs!=null) {
                try {
                    fs.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
        return bitmap;
    }

    /**
     *
     * 讀取圖片屬性:旋轉(zhuǎn)的角度
     * @param path 圖片絕對路徑
     * @return degree旋轉(zhuǎn)的角度
     */

    public static int readPictureDegree(String path) {
        int degree = 0;
        try {
            ExifInterface exifInterface = new ExifInterface(path);
            int orientation = exifInterface.getAttributeInt(
                    ExifInterface.TAG_ORIENTATION,
                    ExifInterface.ORIENTATION_NORMAL);
            switch (orientation) {
                case ExifInterface.ORIENTATION_ROTATE_90:
                    degree = 90;
                    break;
                case ExifInterface.ORIENTATION_ROTATE_180:
                    degree = 180;
                    break;
                case ExifInterface.ORIENTATION_ROTATE_270:
                    degree = 270;
                    break;
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
        return degree;
    }

    /**
     * 調(diào)用native方法
     * @Description:函數(shù)描述
     * @param bit
     * @param quality
     * @param fileName
     * @param optimize
     */
    private static void saveBitmap(Bitmap bit, int quality, String fileName, boolean optimize) {
        compressBitmap(bit, bit.getWidth(), bit.getHeight(), quality, fileName.getBytes(), optimize);
    }

    /**
     * 調(diào)用底層 bitherlibjni.c中的方法
     * @Description:函數(shù)描述
     * @param bit
     * @param w
     * @param h
     * @param quality
     * @param fileNameBytes
     * @param optimize
     * @return
     */
    private static native String compressBitmap(Bitmap bit, int w, int h, int quality, byte[] fileNameBytes,
                                                boolean optimize);
    /**
     * 加載lib下兩個so文件
     */
    static {
        System.loadLibrary("jpegbither");
        System.loadLibrary("bitherjni");
    }
 }

圖片壓縮處理中可能遇到的問題:

  • 請求系統(tǒng)相冊有三個Action
    注意:圖庫(縮略圖) 和 圖片(原圖)
  • ACTION_OPEN_DOCUMENT 僅限4.4或以上使用 默認(rèn)打開原圖
    從圖片獲取到的uri 格式為:content://com.android.providers.media.documents/document/image%666>>>
  • ACTION_GET_CONTENT 4.4以下默認(rèn)打開縮略圖 穆咐。 以上打開文件管理器 供選擇,選擇圖庫打開為縮略圖頁面,選擇圖片打開為原圖瀏覽对湃。
    從圖庫獲取到的uri格式為:content://media/external/images/media/666666
  • ACTION_PICK 都可用崖叫,打開默認(rèn)是縮略圖界面,還需要進(jìn)一步點(diǎn)開查看拍柒。
    參考代碼:
public void pickFromGallery() {
        if (Build.VERSION.SDK_INT < Build.VERSION_CODES.KITKAT) {
            startActivityForResult(new Intent(Intent.ACTION_GET_CONTENT).setType("image/*"),
                    REQUEST_PICK_IMAGE);
        } else {
            Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT);
            intent.addCategory(Intent.CATEGORY_OPENABLE);
            intent.setType("image/*");
            startActivityForResult(intent, REQUEST_KITKAT_PICK_IMAGE);
        }
    }
  • 根據(jù)URI獲取對應(yīng)的文件路徑
    在我們從圖庫中選擇圖片后回調(diào)給我們的data.getData()可能是URI,我們平時(shí)對文件的操作基本上都是基于路徑然后進(jìn)行各種操作與轉(zhuǎn)換心傀,如今我們需要將URI對應(yīng)的文件路徑找出來,具體參考代碼如下:
public static String getPathByUri(Context context, Uri data){
        if (Build.VERSION.SDK_INT < Build.VERSION_CODES.KITKAT) {
            return getPathByUri4BeforeKitkat(context, data);
        }else {
            return getPathByUri4AfterKitkat(context, data);
        }
    }
    //4.4以前通過Uri獲取路徑:data是Uri拆讯,filename是一個String的字符串脂男,用來保存路徑
    public static String getPathByUri4BeforeKitkat(Context context, Uri data) {
        String filename=null;
        if (data.getScheme().toString().compareTo("content") == 0) {
            Cursor cursor = context.getContentResolver().query(data, new String[] { "_data" }, null, null, null);
            if (cursor.moveToFirst()) {
                filename = cursor.getString(0);
            }
        } else if (data.getScheme().toString().compareTo("file") == 0) {// file:///開頭的uri
            filename = data.toString().replace("file://", "");// 替換file://
            if (!filename.startsWith("/mnt")) {// 加上"/mnt"頭
                filename += "/mnt";
            }
        }
        return filename;
    }
    //4.4以后根據(jù)Uri獲取路徑:
    @SuppressLint("NewApi")
    public static String getPathByUri4AfterKitkat(final Context context, final Uri uri) {
        final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;
        // DocumentProvider
        if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {
            if (isExternalStorageDocument(uri)) {// ExternalStorageProvider
                final String docId = DocumentsContract.getDocumentId(uri);
                final String[] split = docId.split(":");
                final String type = split[0];
                if ("primary".equalsIgnoreCase(type)) {
                    return Environment.getExternalStorageDirectory() + "/" + split[1];
                }
            } else if (isDownloadsDocument(uri)) {// DownloadsProvider
                final String id = DocumentsContract.getDocumentId(uri);
                final Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"),
                        Long.valueOf(id));
                return getDataColumn(context, contentUri, null, null);
            } else if (isMediaDocument(uri)) {// MediaProvider
                final String docId = DocumentsContract.getDocumentId(uri);
                final String[] split = docId.split(":");
                final String type = split[0];
                Uri contentUri = null;
                if ("image".equals(type)) {
                    contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
                } else if ("video".equals(type)) {
                    contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
                } else if ("audio".equals(type)) {
                    contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
                }
                final String selection = "_id=?";
                final String[] selectionArgs = new String[] { split[1] };
                return getDataColumn(context, contentUri, selection, selectionArgs);
            }
        } else if ("content".equalsIgnoreCase(uri.getScheme())) {// MediaStore
            // (and
            // general)
            return getDataColumn(context, uri, null, null);
        } else if ("file".equalsIgnoreCase(uri.getScheme())) {// File
            return uri.getPath();
        }
        return null;
    }

    /**
     * Get the value of the data column for this Uri. This is useful for
     * MediaStore Uris, and other file-based ContentProviders.
     *
     * @param context
     *            The context.
     * @param uri
     *            The Uri to query.
     * @param selection
     *            (Optional) Filter used in the query.
     * @param selectionArgs
     *            (Optional) Selection arguments used in the query.
     * @return The value of the _data column, which is typically a file path.
     */
    public static String getDataColumn(Context context, Uri uri, String selection, String[] selectionArgs) {
        Cursor cursor = null;
        final String column = "_data";
        final String[] projection = { column };
        try {
            cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs, null);
            if (cursor != null && cursor.moveToFirst()) {
                final int column_index = cursor.getColumnIndexOrThrow(column);
                return cursor.getString(column_index);
            }
        } finally {
            if (cursor != null)
                cursor.close();
        }
        return null;
    }

    /**
     * @param uri
     *            The Uri to check.
     * @return Whether the Uri authority is ExternalStorageProvider.
     */
    public static boolean isExternalStorageDocument(Uri uri) {
        return "com.android.externalstorage.documents".equals(uri.getAuthority());
    }

    /**
     * @param uri
     *            The Uri to check.
     * @return Whether the Uri authority is DownloadsProvider.
     */
    public static boolean isDownloadsDocument(Uri uri) {
        return "com.android.providers.downloads.documents".equals(uri.getAuthority());
    }

    /**
     * @param uri
     *            The Uri to check.
     * @return Whether the Uri authority is MediaProvider.
     */
    public static boolean isMediaDocument(Uri uri) {
        return "com.android.providers.media.documents".equals(uri.getAuthority());
    }

源碼奉上,自行參考

源碼Github

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末种呐,一起剝皮案震驚了整個濱河市宰翅,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌爽室,老刑警劉巖汁讼,帶你破解...
    沈念sama閱讀 216,402評論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異阔墩,居然都是意外死亡嘿架,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評論 3 392
  • 文/潘曉璐 我一進(jìn)店門戈擒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來眶明,“玉大人,你說我怎么就攤上這事筐高。” “怎么了丑瞧?”我有些...
    開封第一講書人閱讀 162,483評論 0 353
  • 文/不壞的土叔 我叫張陵柑土,是天一觀的道長。 經(jīng)常有香客問我绊汹,道長稽屏,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,165評論 1 292
  • 正文 為了忘掉前任西乖,我火速辦了婚禮狐榔,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘获雕。我一直安慰自己薄腻,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,176評論 6 388
  • 文/花漫 我一把揭開白布届案。 她就那樣靜靜地躺著庵楷,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上尽纽,一...
    開封第一講書人閱讀 51,146評論 1 297
  • 那天咐蚯,我揣著相機(jī)與錄音,去河邊找鬼弄贿。 笑死春锋,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的差凹。 我是一名探鬼主播看疙,決...
    沈念sama閱讀 40,032評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼直奋!你這毒婦竟也來了能庆?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,896評論 0 274
  • 序言:老撾萬榮一對情侶失蹤脚线,失蹤者是張志新(化名)和其女友劉穎搁胆,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體邮绿,經(jīng)...
    沈念sama閱讀 45,311評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡渠旁,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,536評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了船逮。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片顾腊。...
    茶點(diǎn)故事閱讀 39,696評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖挖胃,靈堂內(nèi)的尸體忽然破棺而出杂靶,到底是詐尸還是另有隱情,我是刑警寧澤酱鸭,帶...
    沈念sama閱讀 35,413評論 5 343
  • 正文 年R本政府宣布吗垮,位于F島的核電站,受9級特大地震影響凹髓,放射性物質(zhì)發(fā)生泄漏烁登。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,008評論 3 325
  • 文/蒙蒙 一蔚舀、第九天 我趴在偏房一處隱蔽的房頂上張望饵沧。 院中可真熱鬧,春花似錦赌躺、人聲如沸狼牺。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽锁右。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間咏瑟,已是汗流浹背拂到。 一陣腳步聲響...
    開封第一講書人閱讀 32,815評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留码泞,地道東北人兄旬。 一個月前我還...
    沈念sama閱讀 47,698評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像余寥,于是被迫代替她去往敵國和親领铐。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,592評論 2 353

推薦閱讀更多精彩內(nèi)容