安卓7.0版本升級安裝app產生android.os.FileUriExposedException異常

官方解釋
這裏寫圖片描述

意思大概是:app內可以使用file://這種形式,7.0以後跨應用只能使用content://這種形式,並且要聲明權限.

注意並不僅僅是安裝app需要這樣,想相冊、拍照跨應用的文件共享都需要這樣

解決實例:
在AndroidManifest.xml中application節點裏面添加provider節點中的代碼
這裏寫圖片描述

<provider
            android:name="android.support.v4.content.FileProvider"
            android:authorities="${applicationId}.provider"
            android:exported="false"
            android:grantUriPermissions="true">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/provider_paths"/>
        </provider>

說明:
authorities:app的包名.fileProvider (fileProvider可以隨便寫),上面採用的是gradle的替換直接寫成包名也可以,但是推薦這種方式,多渠道分包的時候不用關心了
grantUriPermissions:必須是true,表示授予 URI 臨時訪問權限
exported:必須是false
resource:中的@xml/file_paths是我們接下來要添加的文件

2、在res目錄下新建一個xml文件夾,並且新建一個provider_paths的xml文件
這裏寫圖片描述

說明:
文件名隨意寫,跟第一步中resource中定義的一樣即可,

3、打開provider_paths.xml文件添加如下內容
這裏寫圖片描述

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path name="external_files" path="."/>
</paths>

說明:
external-path中,
name:可以隨意寫,只是一個名字
path:表示文件路徑,. 表示所有paths下可以有一下節點

<external-cache-path name="name" path="path" />對應Context.getExternalCacheDir()得到的目錄
<external-files-path name="name" path="path" />對應Context#getExternalFilesDir(String) Context.getExternalFilesDir(null)
<external-path name="name" path="path" /> 對應Environment.getExternalStorageDirectory()
<cache-path name="name" path="path" />對應 getCacheDir()得到的目錄
<files-path name="name" path="path" />對應Context.getFilesDir() 目錄

4、修改安裝對應的代碼

//7.0對隱私權限進行了修改 app內如使用的uri不能用intent提供給外部使用 否則會報異常
                Intent intent = new Intent(Intent.ACTION_VIEW);
                if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
                    intent.setDataAndType(Uri.fromFile(new File(BASE_PATH)), "application/vnd.android.package-archive");
                    intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                } else {
                    // 聲明需要的臨時權限
                    // 第二個參數,即第一步中配置的authorities
                    Uri contentUri = FileProvider.getUriForFile(getApplicationContext(), BuildConfig.APPLICATION_ID + ".provider", new File(BASE_PATH));
                    intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                    intent.setDataAndType(contentUri, "application/vnd.android.package-archive");
                }
                startActivity(intent);

說明:
低於7.0的版本還使用老的方式,而高於的要用新的獲取uri的方式。
其中authorities使用的是BuildeConfig獲取的包名,也可以自己寫全,但是推薦使用這種方式,萬一改了包名也不用關心了。
BASE_PATH是自己的文件路徑.

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章