【轉】 android中的文件操做詳解以及內部存儲和外部存儲

摘要 其實安卓文件的操做和Java在pc環境下的操做並沒有二致,之因此須要單獨講解是由於安卓系統提供了不一樣於pc的訪問文件系統根路徑的api,同時對一個應用的私有文件作了統一的管理。根據個人經驗,初學者在這部分感到很容易混淆內部存儲和外部存儲兩個概念。 html

相對路徑和絕對路徑java

在java中,關於相對路徑和絕對路徑是這樣解釋的,若是你很熟悉這部分如下灰色文字能夠跳過:mysql

絕對路徑是指書寫文件的完整路徑,例如d:\java\Hello.java,該路徑中包含文件的完整路徑d:\java以及文件的全名Hello.java。使用該路徑能夠惟一的找到一個文件,不會產生歧義。可是使用絕對路徑在表示文件時,受到的限制很大,且不能在不一樣的操做系統下運行,由於不一樣操做系統下絕對路徑的表達形式存在不一樣。android

相對路徑是指書寫文件的部分路徑,例如\test\Hello.java,該路徑中只包含文件的部分路徑\test和文件的全名Hello.java,部分路徑是指當前路徑下的子路徑,例如當前程序在d:\abc下運行,則該文件的完整路徑就是d:\abc\test。使用這種形式,能夠更加通用的表明文件的位置,使得文件路徑產生必定的靈活性。sql

在Eclipse項目中運行程序時,當前路徑是項目的根目錄,例如工做空間存儲在d:\javaproject,當前項目名稱是Test,則當前路徑是:d:\javaproject\Test。在控制檯下面運行程序時,當前路徑是class文件所在的目錄,若是class文件包含包名,則以該class文件最頂層的包名做爲當前路徑。數據庫

這是java在多數操做系統中這樣操做,很顯然是要咱們儘量的使用相對路徑,可是在安卓中,其實多數狀況下咱們都是使用的絕對路徑。爲何呢?注意上面說到相對路徑是以當前項目所在路徑爲當前路徑,但在安卓中咱們是不可能在項目所在路徑目錄下作任何操做的,由於普通java中咱們的項目建立於服務器(pc也算是服務器),運行於服務器,咱們固然能在服務器操做本身的文件目錄。可是安卓開發中,咱們的項目通常是建立於本身工做的電腦,而運行於手機,既然apk已經運行於手機了,那項目就已經部署到手機上了,應該以apk在手機上的位置來肯定相對路徑,但咱們好像們沒有辦法操做這個路徑的,由於apk是在system目錄下,就算能夠操做,在這個目錄下存取文件也是沒有意義的,好比我寫一個相冊程序,圖片確定是放在外部存儲中,而若是我要保存一個應用的一些設置數據,我是放在內部存儲的data目錄下,所以其實在安卓文件管理中,咱們都是在操做絕對路徑。編程

File類api

操做一個文件(讀寫,建立文件或者目錄)是經過File類來完成的,這個操做和java中徹底一致。服務器

外部存儲external storage和內部存儲internal storageapp

1.內部存儲:

注意內部存儲不是內存。內部存儲位於系統中很特殊的一個位置,若是你想將文件存儲於內部存儲中,那麼文件默認只能被你的應用訪問到,且一個應用所建立的全部文件都在和應用包名相同的目錄下。也就是說應用建立於內部存儲的文件,與這個應用是關聯起來的。當一個應用卸載以後,內部存儲中的這些文件也被刪除。從技術上來說若是你在建立內部存儲文件的時候將文件屬性設置成可讀,其餘app可以訪問本身應用的數據,前提是他知道你這個應用的包名,若是一個文件的屬性是私有(private),那麼即便知道包名其餘應用也沒法訪問。 內部存儲空間十分有限,於是顯得難得,另外,它也是系統自己和系統應用程序主要的數據存儲所在地,一旦內部存儲空間耗盡,手機也就沒法使用了。因此對於內部存儲空間,咱們要儘可能避免使用。Shared Preferences和SQLite數據庫都是存儲在內部存儲空間上的。內部存儲通常用Context來獲取和操做。

getFilesDir()獲取你app的內部存儲空間,至關於你的應用在內部存儲上的根目錄。

若是是要建立一個文件,以下

1
File file = newFile(context.getFilesDir(), filename);

安卓還爲咱們提供了一個簡便方法 openFileOutput()來讀寫應用在內部存儲空間上的文件,下面是一個向文件中寫入文本的例子:

1
2
3
4
5
6
7
8
9
10
String filename =  "myfile" ;
String string =  "Hello world!" ;
FileOutputStream outputStream;
try {
outputStream = openFileOutput(filename, Context.MODE_PRIVATE);
outputStream.write(string.getBytes());
outputStream.close();
catch (Exception e) {
e.printStackTrace();
}

內部存儲的其餘一些操做:

A.列出全部的已建立的文件,這個可能不容易想到,Context竟然有這樣的方法。

1
2
3
4
String[] files = Context.fileList();
for (String file : files) {
Log.e(TAG,  "file is " + file);
}

B.刪除文件,能建立就要可以刪除,固然也會提供了刪除文件的接口,它也很是簡單,只須要提供文件名

1
2
3
4
5
if (Context.deleteFile(filename)) {
Log.e(TAG,  "delete file " + filename +  " sucessfully「);
} else {
Log.e(TAG, " failed to deletefile " + filename);
}

C.建立一個目錄,須要傳入目錄名稱,它返回 一個文件對象用到操做路徑

1
2
File workDir = Context.getDir(dirName, Context.MODE_PRIVATE);
Log.e(TAG,  "workdir " + workDir.getAbsolutePath();


總結一下文件相關操做,能夠得出如下三個特色:
1. 文件操做只須要向函數提供文件名,因此程序本身只須要維護文件名便可;
2. 不用本身去建立文件對象和輸入、輸出流,提供文件名就能夠返回File對象或輸入輸出流
3. 對於路徑操做返回的都是文件對象。

 

2.外部存儲:

最容易混淆的是外部存儲,若是說pc上也要區分出外部存儲和內部存儲的話,那麼自帶的硬盤算是內部存儲,U盤或者移動硬盤算是外部存儲,所以咱們很容易帶着這樣的理解去看待安卓手機,認爲機身固有存儲是內部存儲,而擴展的T卡是外部存儲。好比咱們任務16GB版本的Nexus 4有16G的內部存儲,普通消費者能夠這樣理解,可是安卓的編程中不能,這16GB仍然是外部存儲。

全部的安卓設備都有外部存儲和內部存儲,這兩個名稱來源於安卓的早期設備,那個時候的設備內部存儲確實是固定的,而外部存儲確實是能夠像U盤同樣移動的。可是在後來的設備中,不少中高端機器都將本身的機身存儲擴展到了8G以上,他們將存儲在概念上分紅了"內部internal" 和"外部external" 兩部分,但其實都在手機內部。因此無論安卓手機是否有可移動的sdcard,他們老是有外部存儲和內部存儲。最關鍵的是,咱們都是經過相同的api來訪問可移動的sdcard或者手機自帶的存儲(外部存儲)。

外部存儲雖然概念上有點複雜,但也很好區分,你把手機鏈接電腦,能被電腦識別的部分就必定是外部存儲。

 

關於外部存儲,我以爲api中在介紹Environment.getExternalStorageDirectory()方法的時候說得很清楚:

don't be confused by the word "external" here. This directory can better be thought as media/shared storage. It is a filesystem that can hold a relatively large amount of data and that is shared across all applications (does not enforce permissions). Traditionally this is an SD card, but it may also be implemented as built-in storage in a device that is distinct from the protected internal storage and can be mounted as a filesystem on a computer.

看不懂不要緊,其實跟我說的意思差很少,只是以爲說得比較形象,不知道是個人表述問題,仍是英文在邏輯解釋方面比漢語強,由於白話文實際上是被閹割的漢語。

外部存儲中的文件是能夠被用戶或者其餘應用程序修改的,有兩種類型的文件(或者目錄):

1.公共文件Public files:文件是能夠被自由訪問,且文件的數據對其餘應用或者用戶來講都是由意義的,當應用被卸載以後,其卸載前建立的文件仍然保留。好比camera應用,生成的照片你們都能訪問,並且camera不在了,照片仍然在。

若是你想在外存儲上放公共文件你可使用getExternalStoragePublicDirectory()

1
2
3
4
5
6
7
8
9
public File getAlbumStorageDir(String albumName) {
// Get the directory for the user's public pictures directory.
File file = newFile(Environment.getExternalStoragePublicDirectory(
Environment.DIRECTORY_PICTURES), albumName);
if (!file.mkdirs()) {
Log.e(LOG_TAG,  "Directory not created" );
}
returnfile;
}

在上面的代碼中咱們建立得到了存放picture的目錄,而且新建立一個albumName文件。

若是你的api 版本低於8,那麼不能使用getExternalStoragePublicDirectory(),而是使用Environment.getExternalStorageDirectory(),他不帶參數,也就不能本身建立一個目錄,只是返回外部存儲的根路徑。

2.私有文件Private files:其實因爲是外部存儲的緣由便是是這種類型的文件也能被其餘程序訪問,只不過一個應用私有的文件對其餘應用實際上是沒有訪問價值的(惡意程序除外)。外部存儲上,應用私有文件的價值在於卸載以後,這些文件也會被刪除。相似於內部存儲。

建立應用私有文件的方法是Context.getExternalFilesDir(),以下:

1
2
3
4
5
6
7
8
9
public File getAlbumStorageDir(Context context, String albumName) {
// Get the directory for the app's private pictures directory.
File file = newFile(context.getExternalFilesDir(
Environment.DIRECTORY_PICTURES), albumName);
if (!file.mkdirs()) {
Log.e(LOG_TAG,  "Directory not created" );
}
returnfile;
}

上面的代碼建立了一個picture目錄,並在這個目錄下建立了一個名爲albumName的文件,Environment.DIRECTORY_PICTURES其實就是字符串picture。

全部應用程序的外部存儲的私有文件都放在根目錄的Android/data/下,目錄形式爲/Android/data/<package_name>/

若是你的api 版本低於8,那麼不能使用getExternalFilesDir(),而是使用Environment.getExternalStorageDirectory()得到根路徑以後,本身再想辦法操做/Android/data/<package_name>/下的文件。

也就是說api 8如下的版本在操做文件的時候沒有專門爲私有文件和公共文件的操做提供api支持。你只能先獲取根目錄,而後自行想辦法。

 

在使用外部存儲以前,你必需要先檢查外部存儲的當前狀態,以判斷是否可用。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
boolean mExternalStorageAvailable =  false ;
boolean mExternalStorageWriteable =  false ;
String state = Environment.getExternalStorageState();
if (Environment.MEDIA_MOUNTED.equals(state)) {
// We can read and write the media
mExternalStorageAvailable = mExternalStorageWriteable =  true ;
} elseif(Environment.MEDIA_MOUNTED_READ_ONLY.equals(state)) {
// We can only read the media
mExternalStorageAvailable =  true ;
mExternalStorageWriteable =  false ;
else {
// Something else is wrong. It may be one of many other states, but all we need
//  to know is we can neither read nor write
mExternalStorageAvailable = mExternalStorageWriteable =  false ;
}

 

最後爲了弄清楚getFilesDir,getExternalFilesDir,getExternalStorageDirectory,getExternalStoragePublicDirectory等android文件操做方法,我將這些方法的執行結果打印出來,看看到底路徑是啥樣,在activity中執行如下代碼:

1
2
3
4
5
6
Log.i( "codecraeer" "getFilesDir = "  + getFilesDir());
Log.i( "codecraeer" "getExternalFilesDir = "  + getExternalFilesDir( "exter_test" ).getAbsolutePath());
Log.i( "codecraeer" "getDownloadCacheDirectory = "  + Environment.getDownloadCacheDirectory().getAbsolutePath());
Log.i( "codecraeer" "getDataDirectory = "  + Environment.getDataDirectory().getAbsolutePath());
Log.i( "codecraeer" "getExternalStorageDirectory = "  + Environment.getExternalStorageDirectory().getAbsolutePath());
Log.i( "codecraeer" "getExternalStoragePublicDirectory = "  + Environment.getExternalStoragePublicDirectory( "pub_test" ));

在log中看到以下結果:

從log中咱們能夠看到外部存儲根目錄在我手機(nexus 3)上是/storage/emulated/0,奇怪的是在有些手機上一樣的代碼倒是下面的狀況:

部存儲根目錄爲/mnt/sdcard.

在網上搜了下好像是說三星手機就是這樣。

 

參考文章:http://developer.android.com/training/basics/data-storage/files.html#InternalVsExternalStorage

http://developer.android.com/guide/topics/data/data-storage.html

 

from:http://blog.csdn.net/androidwifi/article/details/17725989/

相關文章
相關標籤/搜索