這篇文章主要為大家展示了Android10的分區(qū)存儲機(jī)制是什么,內(nèi)容簡而易懂,希望大家可以學(xué)習(xí)一下,學(xué)習(xí)完之后肯定會有收獲的,下面讓小編帶大家一起來看看吧。
創(chuàng)新互聯(lián)主要從事成都網(wǎng)站設(shè)計、成都網(wǎng)站建設(shè)、外貿(mào)網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計、企業(yè)做網(wǎng)站、公司建網(wǎng)站等業(yè)務(wù)。立足成都服務(wù)禹王臺,十多年網(wǎng)站建設(shè)經(jīng)驗(yàn),價格優(yōu)惠、服務(wù)專業(yè),歡迎來電咨詢建站服務(wù):18980820575
1. 簡介
大家應(yīng)該都有過這樣的體會,手機(jī)用著用著里面就充斥著各種不懂的文件夾和文件。甚至是連已經(jīng)刪除的軟件的文件夾還存在。
為什么會發(fā)生的這樣的問題呢?
因?yàn)镚oogle的缺席,導(dǎo)致Android生態(tài)野蠻生長,導(dǎo)致很多開發(fā)規(guī)范沒有完全被落實(shí)。
為了解決這樣的問題,Google決定重拳出擊,提出了分區(qū)存儲(Scoped Storage)機(jī)制,也叫沙盒存儲機(jī)制。
那么什么是沙盒存儲機(jī)制呢。
沙盒機(jī)制是一種安全機(jī)制,用于防止應(yīng)用讀取其他應(yīng)用的數(shù)據(jù)。
2. 關(guān)于Android10的分區(qū)機(jī)制
以 Android 10(API 級別 29)及更高版本為目標(biāo)平臺的應(yīng)用在默認(rèn)情況下被賦予了對外部存儲設(shè)備的分區(qū)訪問權(quán)限(即分區(qū)存儲), 對外部存儲文件訪問方式重新設(shè)計,便于用戶更好的管理外部存儲文件。如果不符合條件的會以兼容模式運(yùn)行,兼容模式跟以前一樣,根據(jù)路徑可以直接存儲文件。
應(yīng)用只能看到本應(yīng)用專有的目錄(通過 Context.getExternalFilesDir() 訪問)以及特定類型的媒體。除非您的應(yīng)用需要訪問存放在應(yīng)用的專有目錄以及 MediaStore 之外的文件,否則最好使用分區(qū)存儲。
在發(fā)布Android10的時候官方明確表態(tài):
2020年,主要平臺版本將要求所有應(yīng)用都使用分區(qū)存儲,無論應(yīng)用的目標(biāo) SDK 級別是多少。因此,您應(yīng)該提前確保您的應(yīng)用能夠使用分區(qū)存儲。為此,請確保針對搭載 Android 10(API 級別 29)及更高版本的設(shè)備啟用了該行為。
翻譯成通俗語言,不管是使用requestLegacyExternalStorage=true的方式以兼容模式運(yùn)行還是降低targetSDK都無法在接下來2020年的Android(API 29)10更新中被豁免。
所以為了應(yīng)用的穩(wěn)定性,應(yīng)該盡在進(jìn)行適配。
3. 具體分區(qū)存儲權(quán)限的介紹
默認(rèn)情況下,對于targetSdkVersion大于等于29的應(yīng)用,其訪問權(quán)限范圍限定為分區(qū)存儲。此應(yīng)用無需請求與存儲相關(guān)的用戶權(quán)限,即可以查看外部存儲中以下類型的文件:
分區(qū)存儲將影響在Android10系統(tǒng)首次安裝啟動、且targetSdkVersion >=29的應(yīng)用。需要訪問和共享外部存儲文件的應(yīng)用會受到影響,需要進(jìn)行兼容性適配。
影響范圍:
在Android 10上運(yùn)行的應(yīng)用:
1.targetSdkVersion <= 28,不受影響
2.如果targetSdkVersion >= 29,默認(rèn)情況應(yīng)用外部存儲可見性將被過濾,應(yīng)用需要對分區(qū)存儲進(jìn)行適配。
還有值得注意的是以下兩種情況比較特殊,不會受到分區(qū)存儲的影響:
如果應(yīng)用最先安裝在Android 10以下的系統(tǒng),
1) 然后系統(tǒng)通過Fota升級到Android 10
2) 應(yīng)用通過更新升級到targetSdkVersion >= 29
下面是關(guān)于分區(qū)存儲權(quán)限和其他相關(guān)項(xiàng)目的表格。
類型 | 位置 | 訪問應(yīng)用自己生成的文件 | 訪問其他應(yīng)用生成的的文件 | 訪問方法 | 卸載應(yīng)用是否刪除文件 |
---|---|---|---|---|---|
外部存儲 | Photo/ Video/ Audio/ | 無需權(quán)限 | 需要權(quán)限READ_EXTERNAL_STORAGE | MediaStore Api | 否 |
外部存儲 | Downloads | 無需權(quán)限 | 無需權(quán)限 | 通過存儲訪問框架SAF,加載系統(tǒng)文件選擇器 | 否 |
外部存儲 | 應(yīng)用特定的目錄 | 無需權(quán)限 | 無法直接訪問 | getExternalFilesDir()獲取到屬于應(yīng)用自己的文件路徑 | 是 |
4. 專有目錄存儲
應(yīng)用讀取或?qū)懭霊?yīng)有專有的目錄中的文件時,不需要獲取存儲權(quán)限。
在應(yīng)用中想要獲取當(dāng)前應(yīng)用的專有存儲目錄路徑是可以用Context.getExternalFilesDir()的方式獲取。
val dirpath = context.getExternalFilesDir("") val fileString = dirpath + File.separator val file = File(fileString) ... // 剩下的步驟是用Java IO或者其他IO庫來寫入數(shù)據(jù)
5. 共享媒體集合存儲
在共享媒體集合存儲中保存媒體文件時,需要根據(jù)文件的類型選擇MediaStore。
把相關(guān)數(shù)據(jù)放入到ContentValues中,最后把ContentValues插入到ContentResolver中,并獲得返回的Uri。
通過Uri過得OutputStream,然后用Okio的IO庫,進(jìn)行文件的存儲。
關(guān)于Okio的只是以后有機(jī)會的話,我們再好好講一講。
不要忘了這里需要獲取權(quán)限。
// 把圖片下載到共有媒體集合中,并在相冊中顯示 // 創(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() }
以上就是關(guān)于Android10的分區(qū)存儲機(jī)制是什么的內(nèi)容,如果你們有學(xué)習(xí)到知識或者技能,可以把它分享出去讓更多的人看到。