android增加内置存储分区,详解Android10的分区存储机制(Scoped Storage)适配教程
1. 簡介
大家應(yīng)該都有過這樣的體會(huì),手機(jī)用著用著里面就充斥著各種不懂的文件夾和文件。甚至是連已經(jīng)刪除的軟件的文件夾還存在。
為什么會(huì)發(fā)生的這樣的問題呢?
因?yàn)間oogle的缺席,導(dǎo)致android生態(tài)野蠻生長,導(dǎo)致很多開發(fā)規(guī)范沒有完全被落實(shí)。
為了解決這樣的問題,google決定重拳出擊,提出了分區(qū)存儲(chǔ)(scoped storage)機(jī)制,也叫沙盒存儲(chǔ)機(jī)制。
那么什么是沙盒存儲(chǔ)機(jī)制呢。
沙盒機(jī)制是一種安全機(jī)制,用于防止應(yīng)用讀取其他應(yīng)用的數(shù)據(jù)。
每個(gè)應(yīng)用程序都有自己的存儲(chǔ)空間。
應(yīng)用程序不能翻過自己的目錄,去訪問公共目錄。
應(yīng)用程序請(qǐng)求的數(shù)據(jù)都要通過權(quán)限檢測,不符合要求不會(huì)被放行。
2. 關(guān)于android10的分區(qū)機(jī)制
以 android 10(api 級(jí)別 29)及更高版本為目標(biāo)平臺(tái)的應(yīng)用在默認(rèn)情況下被賦予了對(duì)外部存儲(chǔ)設(shè)備的分區(qū)訪問權(quán)限(即分區(qū)存儲(chǔ)), 對(duì)外部存儲(chǔ)文件訪問方式重新設(shè)計(jì),便于用戶更好的管理外部存儲(chǔ)文件。如果不符合條件的會(huì)以兼容模式運(yùn)行,兼容模式跟以前一樣,根據(jù)路徑可以直接存儲(chǔ)文件。
應(yīng)用只能看到本應(yīng)用專有的目錄(通過 context.getexternalfilesdir() 訪問)以及特定類型的媒體。除非您的應(yīng)用需要訪問存放在應(yīng)用的專有目錄以及 mediastore 之外的文件,否則最好使用分區(qū)存儲(chǔ)。
在發(fā)布android10的時(shí)候官方明確表態(tài):
2020年,主要平臺(tái)版本將要求所有應(yīng)用都使用分區(qū)存儲(chǔ),無論應(yīng)用的目標(biāo) sdk 級(jí)別是多少。因此,您應(yīng)該提前確保您的應(yīng)用能夠使用分區(qū)存儲(chǔ)。為此,請(qǐng)確保針對(duì)搭載 android 10(api 級(jí)別 29)及更高版本的設(shè)備啟用了該行為。
翻譯成通俗語言,不管是使用requestlegacyexternalstorage=true的方式以兼容模式運(yùn)行還是降低targetsdk都無法在接下來2020年的android(api 29)10更新中被豁免。
所以為了應(yīng)用的穩(wěn)定性,應(yīng)該盡在進(jìn)行適配。
3. 具體分區(qū)存儲(chǔ)權(quán)限的介紹
默認(rèn)情況下,對(duì)于targetsdkversion大于等于29的應(yīng)用,其訪問權(quán)限范圍限定為分區(qū)存儲(chǔ)。此應(yīng)用無需請(qǐng)求與存儲(chǔ)相關(guān)的用戶權(quán)限,即可以查看外部存儲(chǔ)中以下類型的文件:
應(yīng)用外部特定目錄中的文件(使用getexternalfilesdir()訪問)。
應(yīng)用自己創(chuàng)建的照片、視頻和音頻(通過mediastore訪問)。
分區(qū)存儲(chǔ)將影響在android10系統(tǒng)首次安裝啟動(dòng)、且targetsdkversion >=29的應(yīng)用。需要訪問和共享外部存儲(chǔ)文件的應(yīng)用會(huì)受到影響,需要進(jìn)行兼容性適配。
影響范圍:
在android 10上運(yùn)行的應(yīng)用:
1.targetsdkversion <= 28,不受影響
2.如果targetsdkversion >= 29,默認(rèn)情況應(yīng)用外部存儲(chǔ)可見性將被過濾,應(yīng)用需要對(duì)分區(qū)存儲(chǔ)進(jìn)行適配。
還有值得注意的是以下兩種情況比較特殊,不會(huì)受到分區(qū)存儲(chǔ)的影響:
如果應(yīng)用最先安裝在android 10以下的系統(tǒng),
1) 然后系統(tǒng)通過fota升級(jí)到android 10
2) 應(yīng)用通過更新升級(jí)到targetsdkversion >= 29
下面是關(guān)于分區(qū)存儲(chǔ)權(quán)限和其他相關(guān)項(xiàng)目的表格。
類型
位置
訪問應(yīng)用自己生成的文件
訪問其他應(yīng)用生成的的文件
訪問方法
卸載應(yīng)用是否刪除文件
外部存儲(chǔ)
photo/ video/ audio/
無需權(quán)限
需要權(quán)限r(nóng)ead_external_storage
mediastore api
否
外部存儲(chǔ)
downloads
無需權(quán)限
無需權(quán)限
通過存儲(chǔ)訪問框架saf,加載系統(tǒng)文件選擇器
否
外部存儲(chǔ)
應(yīng)用特定的目錄
無需權(quán)限
無法直接訪問
getexternalfilesdir()獲取到屬于應(yīng)用自己的文件路徑
是
4. 專有目錄存儲(chǔ)
應(yīng)用讀取或?qū)懭霊?yīng)有專有的目錄中的文件時(shí),不需要獲取存儲(chǔ)權(quán)限。
在應(yīng)用中想要獲取當(dāng)前應(yīng)用的專有存儲(chǔ)目錄路徑是可以用context.getexternalfilesdir()的方式獲取。
val dirpath = context.getexternalfilesdir("")
val filestring = dirpath + file.separator
val file = file(filestring)
... // 剩下的步驟是用java io或者其他io庫來寫入數(shù)據(jù)
5. 共享媒體集合存儲(chǔ)
在共享媒體集合存儲(chǔ)中保存媒體文件時(shí),需要根據(jù)文件的類型選擇mediastore。
把相關(guān)數(shù)據(jù)放入到contentvalues中,最后把contentvalues插入到contentresolver中,并獲得返回的uri。
通過uri過得outputstream,然后用okio的io庫,進(jìn)行文件的存儲(chǔ)。
關(guān)于okio的只是以后有機(jī)會(huì)的話,我們?cè)俸煤弥v一講。
不要忘了這里需要獲取權(quán)限。
// 把圖片下載到共有媒體集合中,并在相冊(cè)中顯示
// 創(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. 其他
到此這篇關(guān)于詳解android10的分區(qū)存儲(chǔ)機(jī)制(scoped storage)適配教程的文章就介紹到這了,更多相關(guān)android10 分區(qū)存儲(chǔ)機(jī)制內(nèi)容請(qǐng)搜索萬仟網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持萬仟網(wǎng)!
總結(jié)
以上是生活随笔為你收集整理的android增加内置存储分区,详解Android10的分区存储机制(Scoped Storage)适配教程的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 坦克炮专栏:U-5TS
- 下一篇: 联想g410怎么设置光盘 如何设置联想G