Android 10 下載圖片/視頻/excel文件等,保存到公共目錄

文章參考

AndroidQ 適配-存儲空間篇
Android Q 要來了,給你一份很"全面"的適配指南!
適配Android Q拍照和讀取相冊圖片
Android-Q適配-存儲方式
Android Q 沙箱適配多媒體文件總結(jié)
AndroidQ 適配-存儲空間篇
Android Q 沙箱適配多媒體文件總結(jié)
Android Q 存儲機(jī)制大變化
Android Q(10) 文件存儲適配

博客內(nèi)容新增

2019-12-20
1搁廓、將公共目錄文件保存在沙盒目錄下面
2020-03-17
1、修改下載文件到公共目錄文件夾下耕皮,不僅僅可以下載Music境蜕、Movies、Pictures凌停、還可以下載普通的excel文件等

問題描述

最近升級到anroid10粱年,google對手機(jī)根目錄文件夾的創(chuàng)建越來越嚴(yán)格了,我倒是覺得這個是一件好事情苦锨,但是逼泣!適配是個頭疼的問題趴泌。我目前處理 android<10 和 >=10 的情況,下載進(jìn)度監(jiān)聽拉庶。
代碼太多嗜憔,只展示部分,剩下代碼請到 github

代碼展示
/**
 * @date: 創(chuàng)建時間:2019/12/11
 * @author: gaoxiaoxiong
 * @descripion:文件操作類的監(jiān)聽
 **/
public interface OnFileDownListener {
  /**
     * @date :2019/12/16 0016
     * @author : gaoxiaoxiong
     * @description:
     * @param status status == -1 表示失敗  status ==0 表示正在下載  status == 1 表示成功
     * @param object 返回成功后的對象參數(shù)
     * @param proGress 當(dāng)前下載百分比
     * @param currentDownProGress 當(dāng)前下載量
     * @param totalProGress 總的量大小
     **/
    void onFileDownStatus(int status,Object object,int proGress, long currentDownProGress, long totalProGress);
}

   /**
     * @date: 2019/5/22 0022
     * @author: gaoxiaoxiong
     * @description: 下載的圖片保存的位置
     **/
    public String getPublickDiskImagePicDir() {
        return getPublickDiskFileDir(BaseApplication.getInstance(), DIRECTORY_PICTURES);
    }


    /**
     * @date: 2019/5/22 0022
     * @author: gaoxiaoxiong
     * @description: 電影保存的位置
     **/
    public String getPublickDiskMoviesDir() {
        return getPublickDiskFileDir(BaseApplication.getInstance(), DIRECTORY_MOVIES);
    }

    /**
     * @date :2019/12/16 0016
     * @author : gaoxiaoxiong
     * @description:音樂保存的位置
     **/
    public String getPublickDiskMusicDir() {
        return getPublickDiskFileDir(BaseApplication.getInstance(), DIRECTORY_MUSIC);
    }

    /**
     * @date 創(chuàng)建時間:2018/12/20
     * @author GaoXiaoXiong
     * @Description: 創(chuàng)建保存圖片的緩存目錄
     */
    public String getPublickDiskImagePicCacheDir() {
        return getPublickDiskCacheDir(BaseApplication.getInstance(), DIRECTORY_PICTURES);
    }

    /**
     * 作者:GaoXiaoXiong
     * 創(chuàng)建時間:2019/1/26
     * 注釋描述:獲取緩存目錄
     *
     * @fileName 獲取外部存儲目錄下緩存的 fileName的文件夾路徑
     */
    public String getPublickDiskCacheDir(Context context, String fileName) {
        String cachePath = null;
        if (Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())
                || !Environment.isExternalStorageRemovable()) {//此目錄下的是外部存儲下的私有的fileName目錄
            cachePath = context.getExternalCacheDir().getPath() + "/" + fileName;  //SDCard/Android/data/你的應(yīng)用包名/cache/fileName
        } else {
            cachePath = context.getCacheDir().getPath() + "/" + fileName;
        }
        File file = new File(cachePath);
        if (!file.exists()) {
            file.mkdirs();
        }
        return file.getAbsolutePath(); //SDCard/Android/data/你的應(yīng)用包名/cache/fileName
    }

    /**
     * @date: 2019/8/2 0002
     * @author: gaoxiaoxiong
     * @description:獲取外部存儲目錄下的 fileName的文件夾路徑
     **/
    public String getPublickDiskFileDir(Context context, String fileName) {
        String cachePath = null;
        if (Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())
                || !Environment.isExternalStorageRemovable()) {//此目錄下的是外部存儲下的私有的fileName目錄
            cachePath = context.getExternalFilesDir(fileName).getAbsolutePath();  //mnt/sdcard/Android/data/com.my.app/files/fileName
        } else {
            cachePath = context.getFilesDir().getPath() + "/" + fileName;        //data/data/com.my.app/files
        }
        File file = new File(cachePath);
        if (!file.exists()) {
            file.mkdirs();
        }
        return file.getAbsolutePath();  //mnt/sdcard/Android/data/com.my.app/files/fileName
    }


    /**
     * @date :2020/3/17 0017
     * @author : gaoxiaoxiong
     * @description:獲取公共目錄氏仗,注意吉捶,只適合android9.0以下的
     **/
    public String getPublickDiskFileDirAndroid9(String fileDir){
        String filePath = null;
        if (Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())
                || !Environment.isExternalStorageRemovable()) {
            filePath = Environment.getExternalStoragePublicDirectory(fileDir).getPath();
        }
        File file = new File(filePath);
        if (!file.exists()) {
            file.mkdirs();
        }
        return file.getAbsolutePath();
    }
   public static final int LOADING = 0;//加載中
   public static final int SUCCESS=1;
   public static final int FAIL=-1;

 /**
     * 如果是要存放到沙盒外部目錄,就需要使用此方法
     * @date: 創(chuàng)建時間:2019/12/11
     * @author: gaoxiaoxiong
     * @descripion: 保存圖片皆尔,視頻呐舔,音樂到公共地區(qū),此操作需要在線程慷蠕,不是我們自己的APP目錄下面的
     * @param downPathUrl 下載文件的路徑珊拼,需要包含后綴
     * @param inserType 存儲類型,可選參數(shù) DIRECTORY_PICTURES  ,DIRECTORY_MOVIES  ,DIRECTORY_MUSIC 流炕,DIRECTORY_DOWNLOADS
     **/
    public void downFileFromServiceToPublicDir(String downPathUrl, Context context, String inserType, OnFileDownListener onFileDownListener) {
        if (inserType.equals(DIRECTORY_DOWNLOADS)){
            if (Build.VERSION.SDK_INT>=29){//android 10
                downUnKnowFileFromService(downPathUrl,context,inserType,onFileDownListener);//返回的是uri
            }else {
                downUnKnowFileFromService(downPathUrl,onFileDownListener);//返回的是file
            }
        }else {
            //下載到沙盒外部公共目錄
            downMusicVideoPicFromService(downPathUrl,context,inserType,onFileDownListener);
        }
    }

 /**
     * 如果是要存放到沙盒外部目錄澎现,就需要使用此方法
     * @date: 創(chuàng)建時間:2019/12/11
     * @author: gaoxiaoxiong
     * @descripion: 下載的文件到 DIRECTORY_DOWNLOADS,只有10以上才有 MediaStore.Downloads
     * @param downPathUrl 下載文件的路徑每辟,需要包含后綴
     * @param inserType 存儲類型 DIRECTORY_DOWNLOADS
     **/
    private void downUnKnowFileFromService(final String downPathUrl,final Context context, String inserType,final OnFileDownListener onFileDownListener){
        if (inserType.equals(DIRECTORY_DOWNLOADS)){
            Observable.just(downPathUrl).subscribeOn(Schedulers.newThread()).map(new Function<String, Uri>() {
                @RequiresApi(api = Build.VERSION_CODES.Q)
                @Override
                public Uri apply(String s) throws Exception {
                    Uri uri = null;
                    try {
                        URL url = new URL(downPathUrl);
                        HttpURLConnection conn = (HttpURLConnection) url.openConnection();
                        conn.setConnectTimeout(30 * 1000);
                        InputStream is = conn.getInputStream();
                        long time = System.currentTimeMillis();
                        int code = conn.getResponseCode();
                        String prefix = downPathUrl.substring(downPathUrl.lastIndexOf(".") + 1);
                        String fileName = null;
                        if (code == HttpURLConnection.HTTP_OK) {
                            fileName = conn.getHeaderField("Content-Disposition");
                            // 通過Content-Disposition獲取文件名剑辫,這點(diǎn)跟服務(wù)器有關(guān),需要靈活變通
                            if (fileName == null || fileName.length() < 1) {
                                // 通過截取URL來獲取文件名
                                URL downloadUrl = conn.getURL(); // 獲得實(shí)際下載文件的URL
                                fileName = downloadUrl.getFile();
                                fileName = fileName.substring(fileName.lastIndexOf("/") + 1);
                            } else {
                                fileName = URLDecoder.decode(fileName.substring(
                                        fileName.indexOf("filename=") + 9), "UTF-8");
                                // 有些文件名會被包含在""里面渠欺,所以要去掉妹蔽,不然無法讀取文件后綴
                                fileName = fileName.replaceAll("\"", "");
                            }
                        }

                        if (isEmpty(fileName)) {
                            fileName = time + "." + prefix;
                        }

                        ContentValues contentValues = new ContentValues();
                        contentValues.put(MediaStore.Downloads.DISPLAY_NAME, fileName);
                        contentValues.put(MediaStore.Downloads.MIME_TYPE,getMIMEType(fileName));
                        contentValues.put(MediaStore.Downloads.DATE_TAKEN, System.currentTimeMillis());
                        uri = context.getContentResolver().insert(MediaStore.Downloads.EXTERNAL_CONTENT_URI, contentValues);
                        BufferedInputStream inputStream = new BufferedInputStream(is);
                        OutputStream os = context.getContentResolver().openOutputStream(uri);
                        if (os != null) {
                            byte[] buffer = new byte[1024];
                            int len;
                            int total = 0;
                            int contentLeng = conn.getContentLength();
                            while ((len = inputStream.read(buffer)) != -1) {
                                os.write(buffer, 0, len);
                                total += len;
                                if (onFileDownListener != null) {
                                    onFileDownListener.onFileDownStatus(LOADING, null, (total * 100 / contentLeng), total, contentLeng);
                                }
                            }
                        }
                        os.flush();
                        inputStream.close();
                        is.close();
                        os.close();
                    } catch (MalformedURLException e) {
                        e.printStackTrace();
                    } catch (UnsupportedEncodingException e) {
                        e.printStackTrace();
                    } catch (FileNotFoundException e) {
                        e.printStackTrace();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                    return uri;
                }
            })
                    .observeOn(AndroidSchedulers.mainThread())
                    .subscribe(new Observer<Uri>() {
                        @Override
                        public void onSubscribe(Disposable d) {

                        }

                        @Override
                        public void onNext(Uri uri) {
                            if (uri != null && onFileDownListener != null) {
                                onFileDownListener.onFileDownStatus(SUCCESS, uri, 0, 0, 0);
                            } else {
                                onFileDownListener.onFileDownStatus(FAIL, null, 0, 0, 0);
                            }
                        }

                        @Override
                        public void onError(Throwable e) {

                        }

                        @Override
                        public void onComplete() {

                        }
                    });
        }
    }


/**
     * 如果是要存放到沙盒外部目錄,就需要使用此方法
     * @date: 創(chuàng)建時間:2019/12/11
     * @author: gaoxiaoxiong
     * @descripion: 保存圖片挠将,視頻胳岂,音樂到公共地區(qū),此操作需要在線程捐名,不是我們自己的APP目錄下面的
     * @param downPathUrl 下載文件的路徑旦万,需要包含后綴
     * @param inserType 存儲類型闹击,可選參數(shù) DIRECTORY_PICTURES  ,DIRECTORY_MOVIES  ,DIRECTORY_MUSIC
     **/
    private void downMusicVideoPicFromService(final String downPathUrl,final Context context,final String inserType,final OnFileDownListener onFileDownListener){
        Observable.just(downPathUrl).subscribeOn(Schedulers.newThread()).map(new Function<String, Uri>() {
            @Override
            public Uri apply(String s) throws Exception {
                Uri uri = null;
                try {
                    URL url = new URL(downPathUrl);
                    HttpURLConnection conn = (HttpURLConnection) url.openConnection();
                    conn.setConnectTimeout(30 * 1000);
                    InputStream is = conn.getInputStream();
                    long time = System.currentTimeMillis();
                    int code = conn.getResponseCode();
                    String prefix = downPathUrl.substring(downPathUrl.lastIndexOf(".") + 1);
                    String fileName = null;
                    if (code == HttpURLConnection.HTTP_OK) {
                        fileName = conn.getHeaderField("Content-Disposition");
                        // 通過Content-Disposition獲取文件名镶蹋,這點(diǎn)跟服務(wù)器有關(guān),需要靈活變通
                        if (fileName == null || fileName.length() < 1) {
                            // 通過截取URL來獲取文件名
                            URL downloadUrl = conn.getURL(); // 獲得實(shí)際下載文件的URL
                            fileName = downloadUrl.getFile();
                            fileName = fileName.substring(fileName.lastIndexOf("/") + 1);
                        } else {
                            fileName = URLDecoder.decode(fileName.substring(
                                    fileName.indexOf("filename=") + 9), "UTF-8");
                            // 有些文件名會被包含在""里面赏半,所以要去掉贺归,不然無法讀取文件后綴
                            fileName = fileName.replaceAll("\"", "");
                        }
                    }

                    if (isEmpty(fileName)) {
                        fileName = time + "." + prefix;
                    }

                    ContentValues contentValues = new ContentValues();
                    if (inserType.equals(DIRECTORY_PICTURES)) {
                        contentValues.put(MediaStore.Images.Media.DISPLAY_NAME, fileName);
                        contentValues.put(MediaStore.Images.Media.MIME_TYPE, getMIMEType(fileName));
                        contentValues.put(MediaStore.Images.Media.DATE_TAKEN, System.currentTimeMillis());
                        //只是往 MediaStore 里面插入一條新的記錄,MediaStore 會返回給我們一個空的 Content Uri
                        //接下來問題就轉(zhuǎn)化為往這個 Content Uri 里面寫入
                        uri = context.getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, contentValues);
                    } else if (inserType.equals(DIRECTORY_MOVIES)) {
                        contentValues.put(MediaStore.Video.Media.MIME_TYPE, getMIMEType(fileName));
                        contentValues.put(MediaStore.Video.Media.DISPLAY_NAME, fileName);
                        contentValues.put(MediaStore.Video.Media.DATE_TAKEN, System.currentTimeMillis());
                        //只是往 MediaStore 里面插入一條新的記錄断箫,MediaStore 會返回給我們一個空的 Content Uri
                        //接下來問題就轉(zhuǎn)化為往這個 Content Uri 里面寫入
                        uri = context.getContentResolver().insert(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, contentValues);
                    } else if (inserType.equals(DIRECTORY_MUSIC)) {
                        contentValues.put(MediaStore.Audio.Media.MIME_TYPE, getMIMEType(fileName));
                        contentValues.put(MediaStore.Audio.Media.DISPLAY_NAME, fileName);
                        if (Build.VERSION.SDK_INT>=29){//android 10
                            contentValues.put(MediaStore.Audio.Media.DATE_TAKEN, System.currentTimeMillis());
                        }
                        //只是往 MediaStore 里面插入一條新的記錄拂酣,MediaStore 會返回給我們一個空的 Content Uri
                        //接下來問題就轉(zhuǎn)化為往這個 Content Uri 里面寫入
                        uri = context.getContentResolver().insert(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, contentValues);
                    }
                    BufferedInputStream inputStream = new BufferedInputStream(is);
                    OutputStream os = context.getContentResolver().openOutputStream(uri);
                    if (os != null) {
                        byte[] buffer = new byte[1024];
                        int len;
                        int total = 0;
                        int contentLeng = conn.getContentLength();
                        while ((len = inputStream.read(buffer)) != -1) {
                            os.write(buffer, 0, len);
                            total += len;
                            if (onFileDownListener != null) {
                                onFileDownListener.onFileDownStatus(LOADING, null, (total * 100 / contentLeng), total, contentLeng);
                            }
                        }
                    }

                    //oppo手機(jī)不會出現(xiàn)在照片里面,但是會出現(xiàn)在圖集里面
                    if (inserType.equals(DIRECTORY_PICTURES)){//如果是圖片
                        //掃描到相冊
                        String[] filePathArray = FileSDCardUtil.getInstance().getPathFromContentUri(uri,context);
                        MediaScannerConnection.scanFile(context, new String[] {filePathArray[0]}, new String[]{"image/jpeg"}, new MediaScannerConnection.OnScanCompletedListener(){
                            @Override
                            public void onScanCompleted(String path, Uri uri) {
                                Log.e(TAG,"PATH:"+path);
                            }
                        } );
                    }
                    os.flush();
                    inputStream.close();
                    is.close();
                    os.close();
                } catch (MalformedURLException e) {
                    e.printStackTrace();
                } catch (UnsupportedEncodingException e) {
                    e.printStackTrace();
                } catch (FileNotFoundException e) {
                    e.printStackTrace();
                } catch (IOException e) {
                    e.printStackTrace();
                }
                return uri;
            }
        })
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Observer<Uri>() {
                    @Override
                    public void onSubscribe(Disposable d) {

                    }

                    @Override
                    public void onNext(Uri uri) {
                        if (uri != null && onFileDownListener != null) {
                            onFileDownListener.onFileDownStatus(SUCCESS, uri, 0, 0, 0);
                        } else {
                            onFileDownListener.onFileDownStatus(FAIL, null, 0, 0, 0);
                        }
                    }

                    @Override
                    public void onError(Throwable e) {
                        Log.e(TAG,"錯誤信息:"+e.getMessage());
                    }

                    @Override
                    public void onComplete() {

                    }
                });
    }
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末仲义,一起剝皮案震驚了整個濱河市婶熬,隨后出現(xiàn)的幾起案子剑勾,更是在濱河造成了極大的恐慌,老刑警劉巖赵颅,帶你破解...
    沈念sama閱讀 216,470評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件虽另,死亡現(xiàn)場離奇詭異,居然都是意外死亡饺谬,警方通過查閱死者的電腦和手機(jī)捂刺,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,393評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來募寨,“玉大人族展,你說我怎么就攤上這事“斡ィ” “怎么了仪缸?”我有些...
    開封第一講書人閱讀 162,577評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長列肢。 經(jīng)常有香客問我腹殿,道長,這世上最難降的妖魔是什么例书? 我笑而不...
    開封第一講書人閱讀 58,176評論 1 292
  • 正文 為了忘掉前任锣尉,我火速辦了婚禮,結(jié)果婚禮上决采,老公的妹妹穿的比我還像新娘自沧。我一直安慰自己,他們只是感情好树瞭,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,189評論 6 388
  • 文/花漫 我一把揭開白布拇厢。 她就那樣靜靜地躺著,像睡著了一般晒喷。 火紅的嫁衣襯著肌膚如雪孝偎。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,155評論 1 299
  • 那天凉敲,我揣著相機(jī)與錄音衣盾,去河邊找鬼。 笑死爷抓,一個胖子當(dāng)著我的面吹牛势决,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播蓝撇,決...
    沈念sama閱讀 40,041評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼果复,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了渤昌?” 一聲冷哼從身側(cè)響起虽抄,我...
    開封第一講書人閱讀 38,903評論 0 274
  • 序言:老撾萬榮一對情侶失蹤走搁,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后迈窟,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體朱盐,經(jīng)...
    沈念sama閱讀 45,319評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,539評論 2 332
  • 正文 我和宋清朗相戀三年菠隆,在試婚紗的時候發(fā)現(xiàn)自己被綠了兵琳。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,703評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡骇径,死狀恐怖躯肌,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情破衔,我是刑警寧澤清女,帶...
    沈念sama閱讀 35,417評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站晰筛,受9級特大地震影響嫡丙,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜读第,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,013評論 3 325
  • 文/蒙蒙 一曙博、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧怜瞒,春花似錦父泳、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,664評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至漾橙,卻和暖如春杆融,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背霜运。 一陣腳步聲響...
    開封第一講書人閱讀 32,818評論 1 269
  • 我被黑心中介騙來泰國打工脾歇, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人觉渴。 一個月前我還...
    沈念sama閱讀 47,711評論 2 368
  • 正文 我出身青樓介劫,卻偏偏與公主長得像徽惋,于是被迫代替她去往敵國和親案淋。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,601評論 2 353