Android存儲訪問及目錄
Android的外部存儲
Android支持外部存儲(case-insensitive filesystem with immutable POSIX permission classes and modes)。
外部存儲可以通過物理介質提供(如SD卡),也可以通過將內部存儲中的一部分封裝而成,設備可以有多個外部存儲實例。
訪問外部存儲的權限
從Android 1.0開始,寫操作受權限WRITE_EXTERNAL_STORAGE保護。
從Android 4.1開始,讀操作受權限READ_EXTERNAL_STORAGE保護。
從Android 4.4開始,應用可以管理在它外部存儲上的特定包名目錄,而不用獲取WRITE_EXTERNAL_STORAGE權限。
比如,一個包名為com.example.foo的應用,可以自由訪問外存上的Android/data/com.example.foo/目錄。
外部存儲對數據提供的保護較少,所以系統不應該存儲敏感數據在外部存儲上。
特別地,配置和log文件應該存儲在內部存儲中,這樣它們可以被有效地保護。
對於多用戶的情況,一般每個用戶都會有自己獨立的外部存儲,應用僅對當前用戶的外部存儲有訪問權限。
Environment API的目錄
getDataDirectory():用戶數據目錄。
getDownloadCacheDirectory():下載緩存內容目錄。
getExternalStorageDirectory():主要的外部存儲目錄。
但是這個目錄很可能當前不能訪問,比如這個目錄被用戶的PC掛載,或者從設備中移除,或者其他問題發生,你可以通過getExternalStorageState()來獲取當前狀態。
還有多用戶或者多外部存儲的情況,此文不再討論。
為了不污染用戶的根命名空間,一般不會直接使用這個外部存儲的根目錄。
任何應用私有的文件的應該被放置在 Context.getExternalFilesDir返回的目錄下,在應用被卸載的時候,系統會清理的就是這個目錄。
另一些共享文件應該被放置在 getExternalStoragePublicDirectory(String)
返回的目錄中。
寫這個路徑需要 WRITE_EXTERNAL_STORAGE
權限,讀需要 READ_EXTERNAL_STORAGE
權限,當然寫權限默認包含了讀權限。
從KITKAT 即Android 4.4開始,如果你的應用只是需要存儲一些內部數據,可以考慮使用 :
getExternalFilesDir(String)
或者getExternalCacheDir(),它們不需要獲取權限。
getExternalStoragePublicDirectory(String type)這個方法接收一個參數,表明目錄所放的文件的類型,傳入的參數是Environment類中的DIRECTORY_XXX靜態變量,比如DIRECTORY_DCIM等。
注意:傳入的類型參數不能是null,返回的目錄路徑有可能不存在,所以必須在使用之前確認一下,比如使用File.mkdirs創建該路徑。
getRootDirectory()得到Android的根目錄。
isExternalStorageEmulated()設備的外存是否是用內存模擬的,是則返回true。(API Level 11)
isExternalStorageRemovable()設備的外存是否是可以拆卸的,比如SD卡,是則返回true。(API Level 9)
Context API中的目錄
getExternalFilesDir(String type)是應用在外部存儲上的目錄。
和Environment類的getExternalStoragePublicDirectory(String type)方法類似,返回包含參數指定的特定類型文件的子目錄。
getExternalCacheDir()是應用的在外部存儲上的緩存目錄。
從Android 4.4這兩個方法不需要讀寫權限,是針對於本應用來說,如果要訪問其他應用的相關目錄,還是需要聲明讀寫權限。
Android 4.4之前的版本要訪問的話還是要聲明讀寫權限的,如果沒有在manifest中寫權限,上面兩個get方法都會返回null。
與上面兩個方法形成對比的是下面兩個方法:
這兩個方法得到的是內存上的目錄。
這些目錄都是屬於應用的,當應用被卸載的時候,里面的內容都會被移除,但是不要依賴於系統的操作。
測試代碼
package com.mengdd.utils.android; import android.content.Context; import android.os.Environment; public class DirectoryUtils { private static final String LOG_TAG = "DirectoryUtils"; public static void getEnvironmentDirectories() { LogUtils.i(LOG_TAG, "getRootDirectory(): " + Environment.getRootDirectory().toString()); LogUtils.i(LOG_TAG, "getDataDirectory(): " + Environment.getDataDirectory().toString()); LogUtils.i(LOG_TAG, "getDownloadCacheDirectory(): " + Environment.getDownloadCacheDirectory().toString()); LogUtils.i(LOG_TAG, "getExternalStorageDirectory(): " + Environment.getExternalStorageDirectory().toString()); LogUtils.i( LOG_TAG, "getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES): " + Environment.getExternalStoragePublicDirectory( Environment.DIRECTORY_PICTURES).toString()); // LogUtils.i( // LOG_TAG, // "isExternalStorageEmulated(): " // + Environment.isExternalStorageEmulated()); // // LogUtils.i( // LOG_TAG, // "isExternalStorageRemovable(): " // + Environment.isExternalStorageRemovable()); } public static void getApplicationDirectories(Context context) { LogUtils.i(LOG_TAG, "context.getFilesDir(): " + context.getFilesDir().toString()); LogUtils.i(LOG_TAG, "context.getCacheDir(): " + context.getCacheDir().toString()); // methods below will return null if the permissions denied LogUtils.i( LOG_TAG, "context.getExternalFilesDir(Environment.DIRECTORY_MOVIES): " + context .getExternalFilesDir(Environment.DIRECTORY_MOVIES)); LogUtils.i( LOG_TAG, "context.getExternalCacheDir(): " + context.getExternalCacheDir()); } }
在MI 2S上輸出Log:
在三星S5660上(API Level 9,注釋掉了兩個方法):
參考資料
android.os.Environment
http://developer.android.com/reference/android/os/Environment.html
External Storage Technical Information
http://source.android.com/devices/tech/storage/
Context
http://developer.android.com/reference/android/content/Context.html