Android 文件存儲

引言:文件存儲[內部存儲]和[外部存儲]。SD 卡上的文件路徑。

時間:2017年06月17日

作者:JustDo23

Github:https://github.com/JustDo23/FastDemo

01. 文件存儲

I/O 流操作來說,Android 和 Java 是基本相同的,區別可能就是文件路徑,相對路徑絕對路徑。某天在瀏覽手機存儲路徑時候產生疑惑和興趣,忽然感覺好久沒有寫過文件操作的代碼,生疏的東西便需要進行梳理。

文件存儲數據存儲中一個重要方式,存儲的路徑不同便有了內部存儲外部存儲

強調:所有的安卓設備都有內部存儲外部存儲

02. 內部存儲

  1. 內部存儲并不是內存。
  2. 內部存儲是系統上一個特殊位置其路徑為/data/data/<package>其中的 package 具體指的是主包名。
  3. 每個應用被安裝之后都會有自己的內部存儲位置,默認情況下只有本應用才能對應的內部存儲文件,當然在創建文件的時候可以指定文件的訪問權限。
  4. 應用被卸載之后,對應的內部存儲也會自動被刪除
  5. 內部存儲空間有限因而顯得可貴,它也是系統本身和系統應用主要的數據存儲所在地,一旦內部存儲空間耗盡,手機也就無法使用了。
  6. 內部存儲一般使用 Context 來獲取和操作。
  7. Shared Preferences 存儲于內部存儲/data/data/<package>/shared_prefs目錄。
  8. SQLite 數據庫存儲于內部存儲/data/data/<package>/database目錄。

03. 外部存儲

  1. 早期的 Android 手機有提供擴展存儲空間的內存卡卡槽,也就是 SD card 存儲,可拆卸可移動類似于 U盤 或者移動硬盤。因而比較容易理解為外部存儲空間。
  2. 現在的 Android 手機基本趨于一體機,取消內存卡卡槽,而在手機機身中內置大容量存儲空間,這部分機身存儲空間同樣是外部存儲。
  3. 直觀區分方法:將手機連接電腦,能被電腦識別出的部分一定是外部存儲。
  4. 剛開始學習 Android 的時候比較熟悉的是/mnt/sdcar目錄,這個目錄指向外部存儲根路徑,現在每次看到的都是/storage/emulated/0目錄,同樣指向外部存儲根路徑,在 DDMS 上可以看到第一個路徑是指向第二個了。
  5. 原則上講外部存儲是公開的,可以被用戶或者其他應用訪問。外部存儲空間可以分為公共文件私有文件兩種類型。
    1. 公共文件就是外部存儲根目錄下的DCIMDownloadMoviesMusicPicturesRingtones等目錄。
    2. 私有文件就是外部存儲根目錄下的Android/data/<package>目錄。
    3. 其實這個所謂的私有文件因為在外部存儲所以同樣是可以被訪問的。
    4. 應用被安裝的時候,同樣會自動生成相應的私有文件目錄
    5. 應用被卸載的時候,相應的私有文件目錄同樣會自動被刪除
    6. 系統媒體掃描程序不會讀取這些私有目錄中的文件,因此不能從MediaStore內容提供程序訪問這些文件。
  6. 外部存儲需要提前在功能清單中申請權限。
  7. 在使用外部存儲之前需要先判斷外部存儲的狀態。每次使用之前都應該先判斷狀態。
  8. 使用的文件目錄不同需要的權限也不大相同。

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. 內部存儲實用方法

  1. 寫入

    /**
     * 向內部存儲寫數據
     */
    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();
      }
    }
    
  2. 讀取

    /**
     * 從內部存儲讀數據
     */
    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();
      }
    }
    
  3. 刪除

    /**
     * 刪除內部存儲上文件
     */
    private void deleteInternalFile() {
      String fileName = "just";// 文件名
      this.deleteFile(fileName);
    }
    

06. 外部存儲多個

這里注意一個情景:一臺設備既有機身內置大容量外部存儲,同時又對外提供了擴展的 SD卡槽,那么外部存儲就有兩個位置了。使用 context.getExternalFilesDir() 默認獲取到內部分區也就是機身外部存儲路徑,那就無法操作 SD卡路徑。不過,從 Android 4.4 開始,通過使用 context.getExternalFilesDirs() 獲取兩個路徑,返回 file 數組。數組中第一個條目被視為是外部主存儲;除非該位置已滿或不可用,否則應該使用該位置。為了向下兼容可以使用支持庫中提供的 ContextCompat.getExternalFilesDirs() 進行兼容,同樣返回數組,但只包含一個目錄。

對于緩存路徑同樣有兼容方法 ContextCompat.getExternalCacheDirs() 向下兼容。

07. 參考文檔

  1. android中的文件操作詳解以及內部存儲和外部存儲
  2. 官網 存儲選項
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • 在Android中我們大多數情況下我們用的都是絕對路徑; 文件存儲位置說明: 內部存儲不是內存,內部存儲位于系統中...
    沈鳳德閱讀 5,457評論 1 2
  • 保存文件 Android 使用與其他平臺上基于磁盤的文件系統類似的文件系統。 本課程講述如何使用 Android ...
    李建彪閱讀 1,159評論 0 2
  • 一直一來沒有認真關注過android 的文件存儲,現在做一個總結,我認為有用的,網上的博客真是寫的叫一個渣渣,根本...
    子丿龍閱讀 387評論 0 0
  • 注0:本文整理于 6 月 13 日注1:今天 6 月 15 日,回過頭來看看,整理了又能怎么樣,還是記不住 And...
    Chenstyle閱讀 549評論 0 1
  • 上一周的周一開始參加閱讀寫作百日活動,第一周過去以后,發現很多事情做起來才有意思,我看到自己的懶惰,拖延,也希望將...
    琳小喵閱讀 218評論 0 0