Android 文件存儲系統原理
存儲的可選項
Android 的文件系統類似於其他平臺的基於磁盤的文件系統。包括以下幾種存儲類別:
- App 私有的存儲空間
- 共享的存儲空間
- Preferences,以 key-value 的形式存儲一些原始的內容,如 SharedPreferences。
- 數據庫
不同的存儲類別扮演瞭不同的角色,比如 App 的私有存儲空間是無法被外部訪問的,可以用來存儲一些 App 私有的敏感信息;共享的存儲空間可以存儲一些所有應用都可以訪問的文件,例如媒體、文檔等可公開的文件;數據庫則是用來處理持久化的;Preferences 用來存儲一些本地字段。
存儲位置的劃分
Android 系統的文件存儲位置劃分為兩大類:內部存儲空間和外部存儲空間 ,其中,外部存儲空間又包括 App 私有目錄和公共目錄。
在大多數設備上,內部存儲空間是小於外部的,但是內部存儲空間是始終可以使用的,與之相反,一些可以插 SD 卡的設備的外部存儲空間是可以拓展的,所以意味著 SD 卡這部分存儲空間不是始終可用的。
內部存儲空間
內部存儲空間是 App 私有的存儲數據的存儲空間,系統會阻止其他應用對這部分數據的訪問,並且在 Android 10(API 級別 29)及更高版本中,系統會對這些位置進行加密。 內部存儲空間的特性讓它很適合存儲隻有 App 本身才能訪問的敏感數據。
內部存儲空間可以通過Context.getFileDir()
或Context.getCacheDir()
獲取到,主要路徑是:
Context.getFileDir() 獲取的路徑為: - data/data/packagename/files (部分手機廠商) - data/user/0/packagename/files (部分手機廠商) Context.getCacheDir() 獲取的路徑為: - data/data/packagename/cache (部分手機廠商) - data/user/0/packagename/cache (部分手機廠商)
外部存儲空間
外部存儲空間包括 App 私有目錄和公共目錄。
- App 私有目錄: App 的私有目錄指其他應用可以訪問當前應用該目錄下的數據,應用卸載後也會隨之刪除。
- 公共目錄:外部可以自由訪問,應用刪除後這部分存儲的數據不會刪除。
App 私有目錄
// 可以通過以下函數獲取 Context.externalCacheDir Context.externalCacheDirs Context.getExternalFilesDir(String) Context.getExternalFilesDirs(String) Context.externalMediaDirs
對應的目錄是:
externalCacheDir: /storage/emulated/0/Android/data/com.chunyu.workdemo/cache externalCacheDirs: /storage/emulated/0/Android/data/com.chunyu.workdemo/cache ExternalFilesDir: /storage/emulated/0/Android/data/com.chunyu.workdemo/files ExternalFilesDirs: /storage/emulated/0/Android/data/com.chunyu.workdemo/files externalMediaDirs: /storage/emulated/0/Android/media/com.chunyu.workdemo
這裡的com.chunyu.workdemo
是 packageName。
外部公共目錄
不要被這裡的“外部”這個詞弄糊塗瞭。最好將此目錄視為媒體/共享的存儲部分。它是一個文件系統,可以保存相對大量的數據,並且在所有應用程序之間共享(不強制執行權限)。傳統上這是一張 SD 卡,但它也可以作為設備中的內置存儲實現,與受保護的內部存儲不同,並且可以作為文件系統安裝在計算機上。
在具有多個用戶的設備上(如 UserManager 所述),每個用戶都有自己的隔離共享存儲。應用程序隻能訪問它們正在運行的用戶的共享存儲。
獲取方式:
Environment.getExternalStorageState() // SD 卡狀態 Environment.getExternalStorageDirectory() Environment.getExternalStoragePublicDirectory(String)
輸出內容:
getExternalStorageState: mounted // 已掛載
getExternalStorageDirectory: /storage/emulated/0
getExternalStoragePublicDirectory: /storage/emulated/0
getExternalStorageDirectory
和getExternalStoragePublicDirectory
已經被標記為棄用,可以使用Context.getExternalFilesDir(String)
、MediaStore 或Intent.ACTION_OPEN_DOCUMENT
等替代方案,它們性能更好。
在上述的需要傳遞 String 參數的方法中,例如Context.getExternalFilesDir(String)
和getExternalStoragePublicDirectory(String)
,String 有以下幾個常量值:
> DIRECTORY_MUSIC // 音樂 > DIRECTORY_PODCASTS // 博客 > DIRECTORY_RINGTONES // 鈴聲 > DIRECTORY_ALARMS // 鬧鐘 > DIRECTORY_NOTIFICATIONS // 通知 > DIRECTORY_PICTURES // 圖片 > DIRECTORY_MOVIES // 電影 > DIRECTORY_DOWNLOADS // 下載 > DIRECTORY_DCIM // 照片 > DIRECTORY_DOCUMENTS // 文檔
不能傳遞空值。
系統目錄
Environment 還提供瞭對一些系統目錄的訪問方法:
Environment.getRootDirectory() // 系統分區的 root 路徑 Environment.getDataDirectory() // 獲取用戶數據目錄的路徑 Environment.getDownloadCacheDirectory() // 獲取用戶緩存目錄的路徑 // 輸出為 getRootDirectory: /system getDataDirectory: /data getDownloadCacheDirectory: /data/cache
清除數據和清除緩存
在 App 中,從上面的方法名中我們也能體會到 cache 和 file 兩種路徑,應該有不同的作用。
清除數據
清除數據清除的是保存在app中所有數據,就是上面提到的位於 packageName 下面的所有文件,包含內部存儲/data/data/packagename/
和外部存儲 /storage/emulated/0/Android/data/packagename/
。
清除緩存
緩存是程序運行時的臨時存儲空間,它可以存放從網絡下載的臨時圖片,從用戶的角度出發清除緩存對用戶並沒有太大的影響,但是清除緩存後用戶再次使用該APP時,由於本地緩存已經被清理,所有的數據需要重新從網絡上獲取。為瞭在清除緩存的時候能夠正常清除與應用相關的緩存,請將緩存文件存放在getCacheDir()
或者getExternalCacheDir()
路徑下。
數據管理權限
某些應用的使用場景時需要廣泛訪問設備上的文件,但無法采用註重隱私保護的存儲最佳做法高效地完成這些操作。對於這些情況,Android 提供瞭一種名為“所有文件訪問權限”的特殊應用訪問權限。 例如,防病毒應用的主要場景可能是需要定期掃描不同目錄中的許多文件。如果此掃描需要反復的用戶交互,讓用戶使用系統文件選擇器選擇目錄,可能就會帶來糟糕的用戶體驗。其他場景(如文件管理器應用、備份和恢復應用以及文檔管理應用)可能也需要考慮類似情況。
應用可以通過執行以下操作向用戶請求“所有文件訪問權限”:
- 在清單中聲明
MANAGE_EXTERNAL_STORAGE
權限。 - 使用
ACTION_MANAGE_ALL_FILES_ACCESS_PERMISSION
設置 Intent ,將用戶引導至一個系統設置頁面,在該頁面上,用戶可以為您的應用啟用以下選項:授予所有文件的管理權限。可以通過Environment.isExternalStorageManager()
方法來檢查是否已獲得這個權限。
MANAGE_EXTERNAL_STORAGE
會授予以下權限:
- 對公共目錄中所有文件的讀寫權限。
- 對
MediaStore.Files
表的內容的訪問權限。 - 除
/Android/data/
、/sdcard/Android
和/sdcard/Android
大多數子目錄外,對所有內部存儲目錄的寫入權限。
獲得此權限的應用仍然無法訪問屬於其他應用的內部存儲空間,因為這些目錄在存儲卷上顯示為 Android/data/ 的子目錄。
到此這篇關於Android 文件存儲系統原理的文章就介紹到這瞭,更多相關Android 文件存儲 內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- Android解決getExternalStorageDirectory在29後廢棄問題(推薦)
- Android調用應用安裝界面方法
- Android11及以上文件讀寫權限申請詳細介紹
- Android 調用設備已有的相機應用詳情
- Android實現調用攝像頭拍照並存儲照片