AndroidQ沙盒機(jī)制之分區(qū)存儲(chǔ)適配
為了讓用戶更好地控制自己的文件,Android Q更改了應(yīng)用訪問(wèn)設(shè)備外部存儲(chǔ)空間中文件的方式。Android Q用更精細(xì)的媒體特定權(quán)限來(lái)替換READ_EXTERNAL_STORAGE和WRITE_EXTERNAL_STORAGE權(quán)限,并且無(wú)需特定權(quán)限,應(yīng)用即可訪問(wèn)自己在外部存儲(chǔ)設(shè)備的文件。
1、針對(duì)應(yīng)用私有文件的隔離存儲(chǔ)沙盒
對(duì)于每個(gè)應(yīng)用,Android Q 都會(huì)創(chuàng)建一個(gè)“隔離存儲(chǔ)沙盒”,以限制其他應(yīng)用訪問(wèn)本應(yīng)用在外部存儲(chǔ)設(shè)備的文件。常見(jiàn)的外部存儲(chǔ)設(shè)備是/sdcard。此定義具有兩個(gè)優(yōu)點(diǎn):
①、需要的權(quán)限更少。 應(yīng)用沙盒中的文件是您應(yīng)用的私有文件。因此,您不再需要任何權(quán)限即可在外部存儲(chǔ)設(shè)備中訪問(wèn)和保存自己的文件;
②、相對(duì)于設(shè)備上的其他應(yīng)用,隱私性更強(qiáng)。 任何其他應(yīng)用都無(wú)法直接訪問(wèn)您應(yīng)用的隔離存儲(chǔ)沙盒中的文件。借助此訪問(wèn)權(quán)限限制,您的應(yīng)用可以更輕松地維護(hù)沙盒文件的隱私性;
在外部存儲(chǔ)設(shè)備存儲(chǔ)文件的最佳位置是Context.getExternalFilesDir()返回文件所在的位置,因此此位置的行為方式在所有Android版本中都保持一致。使用此方法時(shí),需要在媒體環(huán)境中傳遞我們要?jiǎng)?chuàng)建或打開(kāi)的文件類型對(duì)應(yīng)的文件。例如,要保存或訪問(wèn)應(yīng)用私有圖片,請(qǐng)調(diào)用Context.getExternalFilesDir(Environment.DIRECTORY_PICTURES)。
2、媒體文件的共享集合
如果我們的應(yīng)用創(chuàng)建了屬于相應(yīng)用戶的文件,并希望卸載該應(yīng)用時(shí)保留此用戶,則將這些文件保存在某個(gè)通用媒體集合(共享集合)中。共享集合包括:照片、音頻、視頻和下載內(nèi)容。
3、查看其它應(yīng)用的文件所需權(quán)限
我們的應(yīng)用無(wú)需請(qǐng)求任何權(quán)限,即可在這些共享集合中創(chuàng)建和修改自己的文件。但是,我們的應(yīng)用要?jiǎng)?chuàng)建或修改其他應(yīng)用已創(chuàng)建的文件,則必須先請(qǐng)求相應(yīng)權(quán)限:
①、訪問(wèn)照片和視頻共享集合中其他應(yīng)用的文件時(shí),需要 READ_MEDIA_IMAGES 或 READ_MEDIA_VIDEO 權(quán)限(具體取決于您的應(yīng)用需要訪問(wèn)的文件類型);
②、訪問(wèn)音樂(lè)共享集合中其他應(yīng)用的文件時(shí),需要 READ_MEDIA_AUDIO 權(quán)限;
4、訪問(wèn)共享集合
在請(qǐng)求必要的權(quán)限后,我們的應(yīng)用可以使用MediaStore API訪問(wèn)這些集合:
①、對(duì)于照片和視頻共享集合,請(qǐng)使用 MediaStore.Images 或 MediaStore.Video;
②、對(duì)于音樂(lè)共享集合,請(qǐng)使用 MediaStore.Audio;
③、對(duì)于下載內(nèi)容共享集合,請(qǐng)使用 MediaStore.Downloads;
要在原生代碼中訪問(wèn)媒體文件,請(qǐng)使用基于Java或kotlin代碼的MediaStore來(lái)檢索相應(yīng)文件,然后對(duì)相應(yīng)文件描述符傳遞到原生代碼。詳情請(qǐng)參考從原生代碼訪問(wèn)媒體文件部分。
5、保留應(yīng)用在共享集合的文件
默認(rèn)情況下,在用戶卸載應(yīng)用時(shí),Android Q會(huì)清理保存在沙盒的文件。要在卸載應(yīng)用時(shí)保留這些文件,請(qǐng)使用存儲(chǔ)訪問(wèn)框架存儲(chǔ)訪問(wèn)框架,或?qū)⑽募4嬖诠蚕砑现小RA艄蚕砑系奈募?,?qǐng)?jiān)谙嚓P(guān)的MediaStore集合中新插一行,并使用以下方法:
①、至少應(yīng)為 DISPLAY_NAME 和 MIME_TYPE 列提供值;
②、(可選)您可以使用 PRIMARY_DIRECTORY 和 SECONDARY_DIRECTORY 列來(lái)影響文件在磁盤(pán)上的存儲(chǔ)位置;
③、保留 DATA 列不定義。這樣一來(lái),平臺(tái)便可以靈活地將文件保留在沙盒之外;
插入此行后,我們可以使用ContentResolver.openFileDescriptor() 這個(gè)API向文件讀取或?qū)懭霐?shù)據(jù)。
6、訪問(wèn)照片的位置信息
一些照片在Exif元數(shù)據(jù)中包含位置信息,以便用戶查看照片的拍攝地點(diǎn)。由于此位置信息非常敏感,因此默認(rèn)情況下,Android Q會(huì)對(duì)此位置信息進(jìn)行隱藏。如果我們的應(yīng)用需要訪問(wèn)照片的位置信息,需要調(diào)用以下方法:
①、將新的 ACCESS_MEDIA_LOCATION 權(quán)限添加到您應(yīng)用的清單中;
②、在 MediaStore 對(duì)象中,調(diào)用 setRequireOriginal() 并傳入照片的 URI;
Java示例代碼如下:
Uri photoUri = Uri.withAppendedPath( MediaStore.Images.Media.EXTERNAL_CONTENT_URI, cursor.getString(idColumnIndex)); final double[] latLong; if (BuildCompat.isAtLeastQ()) { // When running Android Q, get location data from `ExifInterface`. photoUri = MediaStore.setRequireOriginal(photoUri); InputStream stream = getContentResolver().openInputStream(photoUri); if (stream != null) { ExifInterface exifInterface = new ExifInterface(stream); double[] returnedLatLong = exifInterface.getLatLong(); // If lat/long is null, fall back to the coordinates (0, 0). latLong = returnedLatLong != null ? returnedLatLong : new double[2]; // Don't reuse the stream associated with {@code ExifInterface}. stream.close(); } else { // Failed to load the stream, so return the coordinates (0, 0). latLong = new double[2]; } } else { // On devices running Android 9 (API level 28) and lower, use the // media store columns. latLong = new double[]{ cursor.getFloat(latitudeColumnIndex), cursor.getFloat(longitudeColumnIndex) }; }
kotlin示例代碼如下:
val latLong = if (BuildCompat.isAtLeastQ()) { // When running Android Q, get location data from `ExifInterface`. photoUri = MediaStore.setRequireOriginal(photoUri) contentResolver.openInputStream(photoUri).use { stream -> ExifInterface(stream).run { // If lat/long is null, fall back to the coordinates (0, 0). latLong ?: doubleArrayOf(0.0, 0.0) } } } else { // On devices running Android 9 (API level 28) and lower, use the // media store columns. doubleArrayOf( cursor.getFloat(latitudeColumnIndex).toDouble(), cursor.getFloat(longitudeColumnIndex).toDouble() ) }
7、訪問(wèn)其他應(yīng)用創(chuàng)建的文件
要訪問(wèn)其他應(yīng)用已保存在外部存儲(chǔ)設(shè)備的媒體文件,需要以下步驟:
①、根據(jù)包含您要訪問(wèn)的文件的共享集合請(qǐng)求必要的權(quán)限;
②、使用 ContentResolver 對(duì)象查找并打開(kāi)該文件;
8、向其他應(yīng)用創(chuàng)建的文件寫(xiě)入數(shù)據(jù)
通過(guò)將文件保存在共享集合,我們的應(yīng)用成為該文件的所有者。通常情況下,只有是共享集合的某個(gè)文件所有者時(shí),我們的應(yīng)用才可以向文件寫(xiě)入數(shù)據(jù)。不過(guò),如果我們的應(yīng)用是用戶默認(rèn)的應(yīng)用,我們可以向其他應(yīng)用的文件寫(xiě)入數(shù)據(jù):
①、如果您的應(yīng)用是用戶的默認(rèn)照片管理器應(yīng)用,則可以修改其他應(yīng)用保存到照片和視頻共享集合中的圖片文件;
②、如果您的應(yīng)用是用戶的默認(rèn)音樂(lè)應(yīng)用,則可以修改其他應(yīng)用保存到音樂(lè)共享集合中的音頻文件;
要修改其他應(yīng)用保存在存儲(chǔ)設(shè)備的媒體文件,需要使用ContentResolver找到相應(yīng)文件來(lái)修改。
9、標(biāo)識(shí)特定的外部存儲(chǔ)設(shè)備
在Android 9及以下版本,所有存儲(chǔ)設(shè)備上的所有文件都會(huì)顯示單個(gè)"external"卷名稱。而Android Q為每個(gè)外部存儲(chǔ)設(shè)備提供唯一的卷名稱。此命名系統(tǒng)可以幫助我們高效整理內(nèi)容并且加入索引,還可以控制存儲(chǔ)內(nèi)容的位置。要唯一標(biāo)識(shí)外部存儲(chǔ)設(shè)備的特定文件,我們需要使用卷名稱和ID。例如,主存儲(chǔ)設(shè)備的文件是content://media/external/images/media/12,而命名為FA23-3E92輔助存儲(chǔ)設(shè)備對(duì)應(yīng)文件是content://media/FA23-3E92/images/media/12。
10、獲取外部存儲(chǔ)列表
要獲取所有當(dāng)前可用卷的名稱列表,請(qǐng)調(diào)用 MediaStore.getAllVolumeNames(),如以下代碼段所示:
Set<String> volumeNames = MediaStore.getAllVolumeNames(context);
到此這篇關(guān)于AndroidQ沙盒機(jī)制之分區(qū)存儲(chǔ)適配的文章就介紹到這了,更多相關(guān)AndroidQ 分區(qū)存儲(chǔ)適配內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot整合Mybatis的知識(shí)點(diǎn)匯總
在本篇文章里小編給各位整理的是關(guān)于SpringBoot整合Mybatis的知識(shí)點(diǎn)匯總,有興趣學(xué)習(xí)的參考下。2020-02-02使用Java實(shí)現(xiàn)KMZ和KML數(shù)據(jù)的直接解析
本文主要講解如何用JAVA語(yǔ)言,直接解析KMZ數(shù)據(jù),文章首先介紹google地圖中的KMZ和KML數(shù)據(jù),然后使用代碼的方式實(shí)現(xiàn)數(shù)據(jù)的解析,最后展示解析成果以及如何將數(shù)據(jù)轉(zhuǎn)換成空間WKT數(shù)據(jù),需要的朋友可以參考下2024-06-06Springboot實(shí)現(xiàn)多數(shù)據(jù)源切換詳情
這篇文章主要介紹了Springboot實(shí)現(xiàn)多數(shù)據(jù)源切換詳情,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,感興趣的朋友可以參考一下2022-09-09Springboot獲取前端反饋信息并存入數(shù)據(jù)庫(kù)的實(shí)現(xiàn)代碼
這篇文章主要介紹了Springboot獲取前端反饋信息并存入數(shù)據(jù)庫(kù)的實(shí)現(xiàn)代碼,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03淺談SpringCache與redis集成實(shí)現(xiàn)緩存解決方案
本篇文章主要介紹了淺談SpringCache與redis集成實(shí)現(xiàn)緩存解決方案,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-12-12Java圖形化界面設(shè)計(jì)之容器(JFrame)詳解
這篇文章主要介紹了Java圖形化界面設(shè)計(jì)之容器(JFrame)詳解,條理清晰,依次介紹了Java基本類(JFC),AWT和Swing的區(qū)別,Swing基本框架,圖形化設(shè)計(jì)步驟以及組件容器的使用等相關(guān)內(nèi)容,具有一定參考價(jià)值,需要的朋友可以了解下。2017-11-11利用Sharding-Jdbc進(jìn)行分庫(kù)分表的操作代碼
sharding-jdbc是一個(gè)分布式的關(guān)系型數(shù)據(jù)庫(kù)中間件,今天通過(guò)本文給大家介紹利用Sharding-Jdbc進(jìn)行分庫(kù)分表的操作代碼,代碼簡(jiǎn)單易懂對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2022-01-01