Android7.0適配注意事項
權限更改
- Android6.0引入了動態權限控制(Runtime Permission),Android7.0升級到“私有目錄被限制訪問”即“StrictMode API 政策”。這些更改為用戶帶來了更加安全的操作系統。
權限更改帶來的影響
目錄被限制訪問
- 私有文件的文件權限不在放權給所有的應用,使用
MODE_WORLD_READABLE
或MODE_WORLD_WRITEABLE
進行的操作將觸發 SecurityException。
應對策略:這項權限的變更將意味著你無法通過File API訪問手機存儲上的數據了,基于File API的一些文件瀏覽器等也將受到很大的影響,看到這大家是不是驚呆了呢,不過迄今為止,這種限制尚不能完全執行。 應用仍可能使用原生 API 或 File API 來修改它們的私有目錄權限。 但是,Android官方強烈反對放寬私有目錄的權限。可以看出收起對私有文件的訪問權限是Android將來發展的趨勢。
- 給其他應用傳遞 file:// URI 類型的Uri,可能會導致接受者無法訪問該路徑。 因此,在Android7.0中嘗試傳遞 file:// URI 會觸發 FileUriExposedException。
應對策略:大家可以通過使用FileProvider來解決這一問題。
- DownloadManager 不再按文件名分享私人存儲的文件。COLUMN_LOCAL_FILENAME在Android7.0中被標記為deprecated ,舊版應用在訪問 COLUMN_LOCAL_FILENAME時可能出現無法訪問的路徑。 面向 Android N 或更高版本的應用在嘗試訪問 COLUMN_LOCAL_FILENAME 時會觸發 SecurityException。
應對策略:大家可以通過ContentResolver.openFileDescriptor()來訪問由 DownloadManager 公開的文件。
應用間共享文件
在Android7.0系統上,Android 框架強制執行了 StrictMode API 政策禁止向你的應用外公開 file:// URI。 如果一項包含文件 file:// URI類型 的 Intent 離開你的應用,應用失敗,并出現 FileUriExposedException 異常,如調用系統相機拍照,或裁切照片。
應對策略:若要在應用間共享文件,可以發送 content:// URI類型的Uri,并授予 URI 臨時訪問權限。 進行此授權的最簡單方式是使用 FileProvider類。 如需有關權限和共享文件的更多信息,請參閱共享文件。
解決方案
-
第一步:
- 全局找出項目中,需要修改的地方,如下:
- Uri.parse、Uri.fromFile、file://、content://、Context.getFilesDir()、Environment.getExternalStorageDirectory()、getCacheDir()以及最終要的intent.setDataAndType(為什么需要找這個,因為這個會攜帶uri進行傳遞,這個是重頭戲)
-
第二步:
- 找到罪魁禍首之后,需要按照步驟適配了,依次順序是,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 臨時訪問權限。
- 第四步:
- 為了指定共享的目錄我們需要在資源(res)目錄下創建一個xml目錄,創建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/>國內由于rom眾多,會產生各種路徑,比如華為的/system/media/,以及外置sdcard,
-
第五步:
- 在Java代碼中使用:
//得到緩存路徑的Uri
Uri contentUri = FileProvider.getUriForFile(getActivity(), "com.***.fileprovider", file);
//獲取壁紙
Intent intent = WallpaperManager.getInstance(getActivity()).getCropAndSetWallpaperIntent(contentUri);
//開啟一個Activity顯示圖片,可以將圖片設置為壁紙。調用的是系統的壁紙管理。
getActivity().startActivityForResult(intent, ViewerActivity.REQUEST_CODE_SET_WALLPAPER);
- 需要的權限,intent攜帶的讀寫權限
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
- 在適配過程中,發現有時候addFlag并不能完全的擁有權限,需要grantUriPermission獲取權限
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轉化成適應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轉化成適應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轉化成適應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;
}
}
?