Android7.0適配注意事項
權(quán)限更改
- Android6.0引入了動態(tài)權(quán)限控制(Runtime Permission),Android7.0升級到“私有目錄被限制訪問”即“StrictMode API 政策”。這些更改為用戶帶來了更加安全的操作系統(tǒng)义钉。
權(quán)限更改帶來的影響
目錄被限制訪問
- 私有文件的文件權(quán)限不在放權(quán)給所有的應(yīng)用,使用
MODE_WORLD_READABLE
或MODE_WORLD_WRITEABLE
進行的操作將觸發(fā) SecurityException区匣。
應(yīng)對策略:這項權(quán)限的變更將意味著你無法通過File API訪問手機存儲上的數(shù)據(jù)了箩朴,基于File API的一些文件瀏覽器等也將受到很大的影響,看到這大家是不是驚呆了呢坯癣,不過迄今為止,這種限制尚不能完全執(zhí)行最欠。 應(yīng)用仍可能使用原生 API 或 File API 來修改它們的私有目錄權(quán)限示罗。 但是,Android官方強烈反對放寬私有目錄的權(quán)限芝硬⊙恋悖可以看出收起對私有文件的訪問權(quán)限是Android將來發(fā)展的趨勢。
- 給其他應(yīng)用傳遞 file:// URI 類型的Uri拌阴,可能會導(dǎo)致接受者無法訪問該路徑绍绘。 因此,在Android7.0中嘗試傳遞 file:// URI 會觸發(fā) FileUriExposedException迟赃。
應(yīng)對策略:大家可以通過使用FileProvider來解決這一問題陪拘。
- DownloadManager 不再按文件名分享私人存儲的文件。COLUMN_LOCAL_FILENAME在Android7.0中被標(biāo)記為deprecated 捺氢,舊版應(yīng)用在訪問 COLUMN_LOCAL_FILENAME時可能出現(xiàn)無法訪問的路徑藻丢。 面向 Android N 或更高版本的應(yīng)用在嘗試訪問 COLUMN_LOCAL_FILENAME 時會觸發(fā) SecurityException剪撬。
應(yīng)對策略:大家可以通過ContentResolver.openFileDescriptor()來訪問由 DownloadManager 公開的文件摄乒。
應(yīng)用間共享文件
在Android7.0系統(tǒng)上,Android 框架強制執(zhí)行了 StrictMode API 政策禁止向你的應(yīng)用外公開 file:// URI残黑。 如果一項包含文件 file:// URI類型 的 Intent 離開你的應(yīng)用馍佑,應(yīng)用失敗,并出現(xiàn) FileUriExposedException 異常梨水,如調(diào)用系統(tǒng)相機拍照拭荤,或裁切照片。
應(yīng)對策略:若要在應(yīng)用間共享文件疫诽,可以發(fā)送 content:// URI類型的Uri舅世,并授予 URI 臨時訪問權(quán)限。 進行此授權(quán)的最簡單方式是使用 FileProvider類奇徒。 如需有關(guān)權(quán)限和共享文件的更多信息雏亚,請參閱共享文件。
解決方案
-
第一步:
- 全局找出項目中摩钙,需要修改的地方罢低,如下:
- Uri.parse、Uri.fromFile胖笛、file://网持、content://宜岛、Context.getFilesDir()、Environment.getExternalStorageDirectory()功舀、getCacheDir()以及最終要的intent.setDataAndType(為什么需要找這個萍倡,因為這個會攜帶uri進行傳遞,這個是重頭戲)
-
第二步:
- 找到罪魁禍?zhǔn)字笕砧荆枰凑詹襟E適配了遣铝,依次順序是,AndroidManifest.xml清單文件的修改莉擒,資源文件的修改酿炸,以及Java代碼中的修改
-
第三步:
<!-- 適配Android7.0 FileProvider-->
<manifest>
...
<application>
...
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="package_name.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
</application>
</manifest>
心得:exported:要求必須為false,為true則會報安全異常涨冀。grantUriPermissions:true填硕,表示授予 URI 臨時訪問權(quán)限。
- 第四步:
- 為了指定共享的目錄我們需要在資源(res)目錄下創(chuàng)建一個xml目錄鹿鳖,創(chuàng)建res/xml/filepaths.xml文件
<?xml version="1.0" encoding="utf-8"?>
<resources>
<paths>
<external-path
name="external-path"
path="" />
<files-path
name="files_path"
path="" />
<cache-path
name="cache-path"
path="" />
<root-path
name="root-path"
path="" />
</paths>
</resources>
- <files-path/>代表的根目錄: Context.getFilesDir()
- <external-path/>代表的根目錄: Environment.getExternalStorageDirectory()
- <cache-path/>代表的根目錄: getCacheDir()
- < root-path/>國內(nèi)由于rom眾多扁眯,會產(chǎn)生各種路徑,比如華為的/system/media/翅帜,以及外置sdcard,
-
第五步:
- 在Java代碼中使用:
//得到緩存路徑的Uri
Uri contentUri = FileProvider.getUriForFile(getActivity(), "com.***.fileprovider", file);
//獲取壁紙
Intent intent = WallpaperManager.getInstance(getActivity()).getCropAndSetWallpaperIntent(contentUri);
//開啟一個Activity顯示圖片姻檀,可以將圖片設(shè)置為壁紙。調(diào)用的是系統(tǒng)的壁紙管理涝滴。
getActivity().startActivityForResult(intent, ViewerActivity.REQUEST_CODE_SET_WALLPAPER);
- 需要的權(quán)限绣版,intent攜帶的讀寫權(quán)限
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
- 在適配過程中,發(fā)現(xiàn)有時候addFlag并不能完全的擁有權(quán)限歼疮,需要grantUriPermission獲取權(quán)限
context.grantUriPermission(packageName, uri,
Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
工具類:
public class NougatTools {
private static final String Nougat_FileProvider = "cn.yupaopao.common.fileprovider";
/**
* 將普通uri轉(zhuǎn)化成適應(yīng)7.0的content://形式 針對文件格式
*
* @param context 上下文
* @param file 文件路徑
* @param intent intent
* @param intentType intent.setDataAndType
* @return Intent
*/
public static Intent formatFileProviderIntent(Context context, File file, Intent intent, String intentType) {
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
Uri uri = Uri.fromFile(file);
intent.setDataAndType(uri, intentType);
} else {
Uri uri = FileProvider.getUriForFile(context, Nougat_FileProvider, file);
// 表示文件類型
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
intent.setDataAndType(uri, intentType);
}
return intent;
}
/**
* 將普通uri轉(zhuǎn)化成適應(yīng)7.0的content://形式 針對圖片格式
*
* @param context 上下文
* @param file 文件路徑
* @param intent intent
* @return Intent
*/
public static Intent formatFileProviderPicIntent(Context context, File file, Intent intent) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
Uri uri = FileProvider.getUriForFile(context, Nougat_FileProvider, file);
List<ResolveInfo> resInfoList = context.getPackageManager().queryIntentActivities(
intent, PackageManager.MATCH_DEFAULT_ONLY);
for (ResolveInfo resolveInfo : resInfoList) {
String packageName = resolveInfo.activityInfo.packageName;
context.grantUriPermission(packageName, uri,
Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
}
// 表示圖片類型
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
}
return intent;
}
/**
* 將普通uri轉(zhuǎn)化成適應(yīng)7.0的content://形式
*
* @return Uri
*/
public static Uri formatFileProviderUri(Context context, File file) {
Uri uri;
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
uri = Uri.fromFile(file);
} else {
uri = FileProvider.getUriForFile(context, Nougat_FileProvider, file);
}
return uri;
}
}
?