詳解Android10的分區(qū)存儲機制(Scoped Storage)適配教程
1. 簡介
大家應該都有過這樣的體會,手機用著用著里面就充斥著各種不懂的文件夾和文件。甚至是連已經(jīng)刪除的軟件的文件夾還存在。
為什么會發(fā)生的這樣的問題呢?
因為Google的缺席,導致Android生態(tài)野蠻生長,導致很多開發(fā)規(guī)范沒有完全被落實。
為了解決這樣的問題,Google決定重拳出擊,提出了分區(qū)存儲(Scoped Storage)機制,也叫沙盒存儲機制。
那么什么是沙盒存儲機制呢。
沙盒機制是一種安全機制,用于防止應用讀取其他應用的數(shù)據(jù)。
- 每個應用程序都有自己的存儲空間。
- 應用程序不能翻過自己的目錄,去訪問公共目錄。
- 應用程序請求的數(shù)據(jù)都要通過權限檢測,不符合要求不會被放行。
2. 關于Android10的分區(qū)機制

以 Android 10(API 級別 29)及更高版本為目標平臺的應用在默認情況下被賦予了對外部存儲設備的分區(qū)訪問權限(即分區(qū)存儲), 對外部存儲文件訪問方式重新設計,便于用戶更好的管理外部存儲文件。如果不符合條件的會以兼容模式運行,兼容模式跟以前一樣,根據(jù)路徑可以直接存儲文件。
應用只能看到本應用專有的目錄(通過 Context.getExternalFilesDir() 訪問)以及特定類型的媒體。除非您的應用需要訪問存放在應用的專有目錄以及 MediaStore 之外的文件,否則最好使用分區(qū)存儲。
在發(fā)布Android10的時候官方明確表態(tài):
2020年,主要平臺版本將要求所有應用都使用分區(qū)存儲,無論應用的目標 SDK 級別是多少。因此,您應該提前確保您的應用能夠使用分區(qū)存儲。為此,請確保針對搭載 Android 10(API 級別 29)及更高版本的設備啟用了該行為。
翻譯成通俗語言,不管是使用requestLegacyExternalStorage=true的方式以兼容模式運行還是降低targetSDK都無法在接下來2020年的Android(API 29)10更新中被豁免。
所以為了應用的穩(wěn)定性,應該盡在進行適配。
3. 具體分區(qū)存儲權限的介紹
默認情況下,對于targetSdkVersion大于等于29的應用,其訪問權限范圍限定為分區(qū)存儲。此應用無需請求與存儲相關的用戶權限,即可以查看外部存儲中以下類型的文件:
- 應用外部特定目錄中的文件(使用getExternalFilesDir()訪問)。
- 應用自己創(chuàng)建的照片、視頻和音頻(通過MediaStore訪問)。
分區(qū)存儲將影響在Android10系統(tǒng)首次安裝啟動、且targetSdkVersion >=29的應用。需要訪問和共享外部存儲文件的應用會受到影響,需要進行兼容性適配。
影響范圍:
在Android 10上運行的應用:
1.targetSdkVersion <= 28,不受影響
2.如果targetSdkVersion >= 29,默認情況應用外部存儲可見性將被過濾,應用需要對分區(qū)存儲進行適配。
還有值得注意的是以下兩種情況比較特殊,不會受到分區(qū)存儲的影響:
如果應用最先安裝在Android 10以下的系統(tǒng),
1) 然后系統(tǒng)通過Fota升級到Android 10
2) 應用通過更新升級到targetSdkVersion >= 29
下面是關于分區(qū)存儲權限和其他相關項目的表格。
| 類型 | 位置 | 訪問應用自己生成的文件 | 訪問其他應用生成的的文件 | 訪問方法 | 卸載應用是否刪除文件 |
|---|---|---|---|---|---|
| 外部存儲 | Photo/ Video/ Audio/ | 無需權限 | 需要權限READ_EXTERNAL_STORAGE | MediaStore Api | 否 |
| 外部存儲 | Downloads | 無需權限 | 無需權限 | 通過存儲訪問框架SAF,加載系統(tǒng)文件選擇器 | 否 |
| 外部存儲 | 應用特定的目錄 | 無需權限 | 無法直接訪問 | getExternalFilesDir()獲取到屬于應用自己的文件路徑 | 是 |
4. 專有目錄存儲
應用讀取或寫入應有專有的目錄中的文件時,不需要獲取存儲權限。
在應用中想要獲取當前應用的專有存儲目錄路徑是可以用Context.getExternalFilesDir()的方式獲取。
val dirpath = context.getExternalFilesDir("")
val fileString = dirpath + File.separator
val file = File(fileString)
... // 剩下的步驟是用Java IO或者其他IO庫來寫入數(shù)據(jù)
5. 共享媒體集合存儲
在共享媒體集合存儲中保存媒體文件時,需要根據(jù)文件的類型選擇MediaStore。
把相關數(shù)據(jù)放入到ContentValues中,最后把ContentValues插入到ContentResolver中,并獲得返回的Uri。
通過Uri過得OutputStream,然后用Okio的IO庫,進行文件的存儲。
關于Okio的只是以后有機會的話,我們再好好講一講。
不要忘了這里需要獲取權限。
// 把圖片下載到共有媒體集合中,并在相冊中顯示
// 創(chuàng)建ContentValues, 并加入信息
val values = ContentValues()
values.put(MediaStore.Images.Media.DESCRIPTION, downloadedFile.name)
values.put(MediaStore.Images.Media.DISPLAY_NAME, downloadedFile.name)
values.put(MediaStore.Images.Media.MIME_TYPE, mimeType)
values.put(MediaStore.Images.Media.TITLE, downloadedFile.name)
values.put(
MediaStore.Images.Media.RELATIVE_PATH,
"${Environment.DIRECTORY_PICTURES}/${downloadedFile.name}"
)
// 插入到ContentResolver,并返回Uri
val insertUri = context.contentResolver.insert(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
values
)
if (insertUri != null) {
// 獲取OutputStream
val outputStream = context.contentResolver.openOutputStream(insertUri)
if (outputStream != null) {
sink = outputStream.sink().buffer()
} else {
return@runCatching FileDownloadResult.OthersError
}
} else {
return@runCatching FileDownloadResult.OthersError
}
val responseBody = response.body ?: return@runCatching FileDownloadResult.OthersError
try {
val contentLength = responseBody.contentLength()
if (contentLength > FileUtil.getAvailableSize(dirPath)) {
continuation.resume(FileDownloadResult.StorageError)
}
var totalRead: Long = 0
var lastRead: Long
do {
lastRead = responseBody.source().read(sink.buffer(), BUFFER_SIZE)
if (lastRead == -1L) {
break
}
totalRead += lastRead
sink.emitCompleteSegments()
} while (true)
sink.writeAll(responseBody.source())
sink.close()
responseBody.close()
}
6. 其他
Github: https://github.com/HyejeanMOON/ScopedStorageDemo
到此這篇關于詳解Android10的分區(qū)存儲機制(Scoped Storage)適配教程的文章就介紹到這了,更多相關Android10 分區(qū)存儲機制內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Android中隱藏狀態(tài)欄和標題欄的方法匯總(隱藏狀態(tài)欄、標題欄的五種方法)
這篇文章主要介紹了Android中隱藏狀態(tài)欄和標題欄的方法匯總(隱藏狀態(tài)欄、標題欄的五種方法),非常不錯,具有參考借鑒價值,需要的朋友可以參考下2017-02-02
android自動安裝apk代碼實例(不使用apk安裝器安裝)
這篇文章主要介紹了android自動安裝apk代碼實例,代碼簡單,大家參考使用吧2013-11-11
Android頂部狀態(tài)欄透明化并釋放空間的兩種實現(xiàn)方法
這篇文章主要介紹了Android頂部狀態(tài)欄透明化并釋放空間的兩種實現(xiàn)方法,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2017-03-03
使用RoundedBitmapDrawable生成圓角圖片的方法
由于RoundedBitmapDrawable類沒有直接提供生成圓形圖片的方法,所以生成圓形圖片首先需要對原始圖片進行裁剪,將圖片裁剪成正方形,最后再生成圓形圖片,具體實現(xiàn)方法,可以參考下本文2016-09-09
android中寫一個內部類來選擇文件夾中指定的圖片類型實例說明
選擇文件夾中指定的圖片類型,本類是用來選擇文件夾中是.jpg類型的圖片具體實現(xiàn)如下,感興趣的朋友可以參考下哈2013-06-06

