引言:文件存儲[內部存儲]和[外部存儲]。SD 卡上的文件路徑。
時間:2017年06月17日
作者:JustDo23
01. 文件存儲
就 I/O 流操作來說,Android 和 Java 是基本相同的,區別可能就是文件路徑,相對路徑與絕對路徑。某天在瀏覽手機存儲路徑時候產生疑惑和興趣,忽然感覺好久沒有寫過文件操作的代碼,生疏的東西便需要進行梳理。
文件存儲是數據存儲中一個重要方式,存儲的路徑不同便有了內部存儲和外部存儲。
強調:所有的安卓設備都有內部存儲和外部存儲。
02. 內部存儲
- 內部存儲并不是內存。
- 內部存儲是系統上一個特殊位置其路徑為
/data/data/<package>
其中的 package 具體指的是主包名。 - 每個應用被安裝之后都會有自己的內部存儲位置,默認情況下只有本應用才能對應的內部存儲文件,當然在創建文件的時候可以指定文件的訪問權限。
- 當應用被卸載之后,對應的內部存儲也會自動被刪除。
- 內部存儲空間有限因而顯得可貴,它也是系統本身和系統應用主要的數據存儲所在地,一旦內部存儲空間耗盡,手機也就無法使用了。
- 內部存儲一般使用 Context 來獲取和操作。
-
Shared Preferences 存儲于內部存儲
/data/data/<package>/shared_prefs
目錄。 -
SQLite 數據庫存儲于內部存儲
/data/data/<package>/database
目錄。
03. 外部存儲
- 早期的 Android 手機有提供擴展存儲空間的內存卡卡槽,也就是 SD card 存儲,可拆卸可移動類似于 U盤 或者移動硬盤。因而比較容易理解為外部存儲空間。
- 現在的 Android 手機基本趨于一體機,取消內存卡卡槽,而在手機機身中內置大容量存儲空間,這部分機身存儲空間同樣是外部存儲。
- 直觀區分方法:將手機連接電腦,能被電腦識別出的部分一定是外部存儲。
- 剛開始學習 Android 的時候比較熟悉的是
/mnt/sdcar
目錄,這個目錄指向外部存儲根路徑,現在每次看到的都是/storage/emulated/0
目錄,同樣指向外部存儲根路徑,在 DDMS 上可以看到第一個路徑是指向第二個了。 - 原則上講外部存儲是公開的,可以被用戶或者其他應用訪問。外部存儲空間可以分為公共文件和私有文件兩種類型。
-
公共文件就是
外部存儲根目錄
下的DCIM
,Download
,Movies
,Music
,Pictures
,Ringtones
等目錄。 -
私有文件就是
外部存儲根目錄
下的Android/data/<package>
目錄。 - 其實這個所謂的私有文件因為在外部存儲所以同樣是可以被訪問的。
- 當應用被安裝的時候,同樣會自動生成相應的私有文件目錄。
- 當應用被卸載的時候,相應的私有文件目錄同樣會自動被刪除。
- 系統媒體掃描程序不會讀取這些私有目錄中的文件,因此不能從MediaStore內容提供程序訪問這些文件。
-
公共文件就是
- 外部存儲需要提前在功能清單中申請權限。
- 在使用外部存儲之前需要先判斷外部存儲的狀態。每次使用之前都應該先判斷狀態。
- 使用的文件目錄不同需要的權限也不大相同。

04. 文件路徑
通過代碼查看相關路徑
/**
* 查看文件路徑
*
* @param context
*/
private void filePath(Context context) {
// 通過 Context 獲取內部存儲路徑
LogUtils.e("context.getFilesDir() = " + context.getFilesDir());// [ /data/data/com.just.fast/files ]
LogUtils.e("context.getCacheDir() = " + context.getCacheDir());// [ /data/data/com.just.fast/cache ]
LogUtils.e("context.getDir(dirName, MODE_PRIVATE) = " + context.getDir("dirName", MODE_PRIVATE));// [ /data/data/com.just.fast/app_dirName ]
// 獲取外部存儲根路徑[先判斷外部存儲是否掛載][讀寫文件需要權限]
LogUtils.e("Environment.getExternalStorageState() = " + Environment.getExternalStorageState());// [ mounted ]
LogUtils.e("Environment.getExternalStorageDirectory() = " + Environment.getExternalStorageDirectory());// [ /storage/emulated/0 ]
// [不建議在外部存儲根目錄操作][因此獲取外部存儲私有路徑][不需要權限][私有文件]
LogUtils.e("context.getExternalFilesDir(null) = " + context.getExternalFilesDir(null));// [ /storage/emulated/0/Android/data/com.just.fast/files ]
LogUtils.e("context.getExternalFilesDir(Environment.DIRECTORY_MOVIES) = " + context.getExternalFilesDir(Environment.DIRECTORY_MOVIES));// [ /storage/emulated/0/Android/data/com.just.fast/files/Movies ]
LogUtils.e("context.getExternalCacheDir() = " + context.getExternalCacheDir());// [ /storage/emulated/0/Android/data/com.just.fast/cache ]
// [不建議在外部存儲根目錄操作][因此獲取外部存儲共享路徑][讀寫文件需要權限][公共文件]
LogUtils.e("Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_MUSIC) = " + Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_MUSIC));// [ /storage/emulated/0/Music ]
// 其他路徑[獲取系統根路徑下的相關路徑]
LogUtils.e("Environment.getRootDirectory() = " + Environment.getRootDirectory());// [ /system ]
LogUtils.e("Environment.getDataDirectory() = " + Environment.getDataDirectory());// [ /data ]
LogUtils.e("Environment.getDownloadCacheDirectory() = " + Environment.getDownloadCacheDirectory());// [ /cache ]
// [true,仿真外部存儲][一體機之類]
LogUtils.e("Environment.isExternalStorageEmulated() = " + Environment.isExternalStorageEmulated());// [ true ]
// [true,外部存儲物理硬件可以移除][SD卡子類][false,一體機之類]
LogUtils.e("Environment.isExternalStorageRemovable() = " + Environment.isExternalStorageRemovable());// [ false ]
}
代碼注釋多精簡結果
// 通過 Context 獲取內部存儲路徑
context.getFilesDir() = [ /data/data/com.just.fast/files ]
context.getCacheDir() = [ /data/data/com.just.fast/cache ]
context.getDir(dirName, MODE_PRIVATE) = [ /data/data/com.just.fast/app_dirName ]
// 獲取外部存儲根路徑[先判斷外部存儲是否掛載][讀寫文件需要權限]
Environment.getExternalStorageState() = [ mounted ]
Environment.getExternalStorageDirectory() = [ /storage/emulated/0 ]
// [不建議在外部存儲根目錄操作][因此獲取外部存儲私有路徑][不需要權限][私有文件]
context.getExternalFilesDir(null) = [ /storage/emulated/0/Android/data/com.just.fast/files ]
context.getExternalFilesDir(Environment.DIRECTORY_MOVIES) = [ /storage/emulated/0/Android/data/com.just.fast/files/Movies ]
context.getExternalCacheDir() = [ /storage/emulated/0/Android/data/com.just.fast/cache ]
// [不建議在外部存儲根目錄操作][因此獲取外部存儲共享路徑][讀寫文件需要權限][公共文件]
Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_MUSIC) = [ /storage/emulated/0/Music ]
// 其他路徑[獲取系統根路徑下的相關路徑]
Environment.getRootDirectory() = [ /system ]
Environment.getDataDirectory() = [ /data ]
Environment.getDownloadCacheDirectory() = [ /cache ]
// [true,仿真外部存儲][一體機之類]
Environment.isExternalStorageEmulated() = [ true ]
// [true,外部存儲物理硬件可以移除][SD卡子類][false,一體機之類]
Environment.isExternalStorageRemovable() = [ false ]
05. 內部存儲實用方法
-
寫入
/** * 向內部存儲寫數據 */ private void write2Internal() { String fileName = "just";// 文件名 String writeContent = "Test open and write file internal.";// 寫入內容 try { FileOutputStream fileOutputStream = this.openFileOutput(fileName, MODE_APPEND);// 打開文件輸出流 fileOutputStream.write(writeContent.getBytes());// 流寫入數據 fileOutputStream.close();// 關閉流 } catch (IOException e) { e.printStackTrace(); } }
-
讀取
/** * 從內部存儲讀數據 */ private void readFromInternal() { String fileName = "just";// 文件名 try { FileInputStream fileInputStream = this.openFileInput(fileName);// 打開文件輸入流 byte[] allByte = new byte[fileInputStream.available()];// 字節數組 fileInputStream.read(allByte);// 讀取 String readContent = new String(allByte);// 轉換為字符串 fileInputStream.close(); tv_read.setText(readContent); } catch (IOException e) { e.printStackTrace(); } }
-
刪除
/** * 刪除內部存儲上文件 */ private void deleteInternalFile() { String fileName = "just";// 文件名 this.deleteFile(fileName); }
06. 外部存儲多個
這里注意一個情景:一臺設備既有機身內置大容量外部存儲,同時又對外提供了擴展的 SD卡槽,那么外部存儲就有兩個位置了。使用 context.getExternalFilesDir()
默認獲取到內部分區也就是機身外部存儲路徑,那就無法操作 SD卡路徑。不過,從 Android 4.4 開始,通過使用 context.getExternalFilesDirs()
獲取兩個路徑,返回 file 數組。數組中第一個條目被視為是外部主存儲;除非該位置已滿或不可用,否則應該使用該位置。為了向下兼容可以使用支持庫中提供的 ContextCompat.getExternalFilesDirs()
進行兼容,同樣返回數組,但只包含一個目錄。
對于緩存路徑同樣有兼容方法 ContextCompat.getExternalCacheDirs()
向下兼容。