其實(shí)安卓文件的操作和java在pc環(huán)境下的操作并無(wú)二致,之所以需要單獨(dú)講解是因?yàn)榘沧肯到y(tǒng)提供了不同于pc的訪問(wèn)文件系統(tǒng)根路徑的api,同時(shí)對(duì)一個(gè)應(yīng)用的私有文件做了統(tǒng)一的管理。根據(jù)我的經(jīng)驗(yàn),初學(xué)者在這部分感到很容易混淆內(nèi)部存儲(chǔ)和外部存儲(chǔ)兩個(gè)概念。
相對(duì)路徑和絕對(duì)路徑
在java中,關(guān)于相對(duì)路徑和絕對(duì)路徑是這樣解釋的,如果你很熟悉這部分以下灰色文字可以跳過(guò):
絕對(duì)路徑是指書(shū)寫(xiě)文件的完整路徑,例如d:\java\Hello.java,該路徑中包含文件的完整路徑d:\java以及文件的全名Hello.java。使用該路徑可以唯一的找到一個(gè)文件,不會(huì)產(chǎn)生歧義。但是使用絕對(duì)路徑在表示文件時(shí),受到的限制很大,且不能在不同的操作系統(tǒng)下運(yùn)行,因?yàn)椴煌僮飨到y(tǒng)下絕對(duì)路徑的表達(dá)形式存在不同。
相對(duì)路徑是指書(shū)寫(xiě)文件的部分路徑,例如\test\Hello.java,該路徑中只包含文件的部分路徑\test和文件的全名Hello.java,部分路徑是指當(dāng)前路徑下的子路徑,例如當(dāng)前程序在d:\abc下運(yùn)行,則該文件的完整路徑就是d:\abc\test。使用這種形式,可以更加通用的代表文件的位置,使得文件路徑產(chǎn)生一定的靈活性。
在Eclipse項(xiàng)目中運(yùn)行程序時(shí),當(dāng)前路徑是項(xiàng)目的根目錄,例如工作空間存儲(chǔ)在d:\javaproject,當(dāng)前項(xiàng)目名稱(chēng)是Test,則當(dāng)前路徑是:d:\javaproject\Test。在控制臺(tái)下面運(yùn)行程序時(shí),當(dāng)前路徑是class文件所在的目錄,如果class文件包含包名,則以該class文件最頂層的包名作為當(dāng)前路徑。
這是java在多數(shù)操作系統(tǒng)中這樣操作,很顯然是要我們盡可能的使用相對(duì)路徑,但是在安卓中,其實(shí)多數(shù)情況下我們都是使用的絕對(duì)路徑。為什么呢?注意上面說(shuō)到相對(duì)路徑是以當(dāng)前項(xiàng)目所在路徑為當(dāng)前路徑,但在安卓中我們是不可能在項(xiàng)目所在路徑目錄下做任何操作的,因?yàn)槠胀╦ava中我們的項(xiàng)目創(chuàng)建于服務(wù)器(pc也算是服務(wù)器),運(yùn)行于服務(wù)器,我們當(dāng)然能在服務(wù)器操作自己的文件目錄。但是安卓開(kāi)發(fā)中,我們的項(xiàng)目一般是創(chuàng)建于自己工作的電腦,而運(yùn)行于手機(jī),既然apk已經(jīng)運(yùn)行于手機(jī)了,那項(xiàng)目就已經(jīng)部署到手機(jī)上了,應(yīng)該以apk在手機(jī)上的位置來(lái)確定相對(duì)路徑,但我們好像們沒(méi)有辦法操作這個(gè)路徑的,因?yàn)閍pk是在system目錄下,就算可以操作,在這個(gè)目錄下存取文件也是沒(méi)有意義的,比如我寫(xiě)一個(gè)相冊(cè)程序,圖片肯定是放在外部存儲(chǔ)中,而如果我要保存一個(gè)應(yīng)用的一些設(shè)置數(shù)據(jù),我是放在內(nèi)部存儲(chǔ)的data目錄下,因此其實(shí)在安卓文件管理中,我們都是在操作絕對(duì)路徑。
File類(lèi)
操作一個(gè)文件(讀寫(xiě),創(chuàng)建文件或者目錄)是通過(guò)File類(lèi)來(lái)完成的,這個(gè)操作和java中完全一致。
外部存儲(chǔ)external storage和內(nèi)部存儲(chǔ)internalstorage
1.內(nèi)部存儲(chǔ):
注意內(nèi)部存儲(chǔ)不是內(nèi)存。內(nèi)部存儲(chǔ)位于系統(tǒng)中很特殊的一個(gè)位置,如果你想將文件存儲(chǔ)于內(nèi)部存儲(chǔ)中,那么文件默認(rèn)只能被你的應(yīng)用訪問(wèn)到,且一個(gè)應(yīng)用所創(chuàng)建的所有文件都在和應(yīng)用包名相同的目錄下。也就是說(shuō)應(yīng)用創(chuàng)建于內(nèi)部存儲(chǔ)的文件,與這個(gè)應(yīng)用是關(guān)聯(lián)起來(lái)的。當(dāng)一個(gè)應(yīng)用卸載之后,內(nèi)部存儲(chǔ)中的這些文件也被刪除。從技術(shù)上來(lái)講如果你在創(chuàng)建內(nèi)部存儲(chǔ)文件的時(shí)候?qū)⑽募傩栽O(shè)置成可讀,其他app能夠訪問(wèn)自己應(yīng)用的數(shù)據(jù),前提是他知道你這個(gè)應(yīng)用的包名,如果一個(gè)文件的屬性是私有(private),那么即使知道包名其他應(yīng)用也無(wú)法訪問(wèn)。內(nèi)部存儲(chǔ)空間十分有限,因而顯得可貴,另外,它也是系統(tǒng)本身和系統(tǒng)應(yīng)用程序主要的數(shù)據(jù)存儲(chǔ)所在地,一旦內(nèi)部存儲(chǔ)空間耗盡,手機(jī)也就無(wú)法使用了。所以對(duì)于內(nèi)部存儲(chǔ)空間,我們要盡量避免使用。Shared Preferences和SQLite數(shù)據(jù)庫(kù)都是存儲(chǔ)在內(nèi)部存儲(chǔ)空間上的。內(nèi)部存儲(chǔ)一般用Context來(lái)獲取和操作。
getFilesDir()獲取你app的內(nèi)部存儲(chǔ)空間,相當(dāng)于你的應(yīng)用在內(nèi)部存儲(chǔ)上的根目錄。
如果是要?jiǎng)?chuàng)建一個(gè)文件,如下
1
File file = newFile(context.getFilesDir(), filename);
安卓還為我們提供了一個(gè)簡(jiǎn)便方法openFileOutput()來(lái)讀寫(xiě)應(yīng)用在內(nèi)部存儲(chǔ)空間上的文件,下面是一個(gè)向文件中寫(xiě)入文本的例子:
1
2
3
4
5
6
7
8
9
10String 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();
}
內(nèi)部存儲(chǔ)的其他一些操作:
A.列出所有的已創(chuàng)建的文件,這個(gè)可能不容易想到,Context居然有這樣的方法。
1
2
3
4String[] files = Context.fileList();
for(String file : files) {
Log.e(TAG,"file is "+ file);
}
B.刪除文件,能創(chuàng)建就要能夠刪除,當(dāng)然也會(huì)提供了刪除文件的接口,它也非常簡(jiǎn)單,只需要提供文件名
1
2
3
4
5if(Context.deleteFile(filename)) {
Log.e(TAG,"delete file "+ filename +" sucessfully“);
} else {
Log.e(TAG, "failed to deletefile " + filename);
}
C.創(chuàng)建一個(gè)目錄,需要傳入目錄名稱(chēng),它返回 一個(gè)文件對(duì)象用到操作路徑
1
2File workDir = Context.getDir(dirName, Context.MODE_PRIVATE);
Log.e(TAG,"workdir "+ workDir.getAbsolutePath();
總結(jié)一下文件相關(guān)操作,可以得出以下三個(gè)特點(diǎn):
1. 文件操作只需要向函數(shù)提供文件名,所以程序自己只需要維護(hù)文件名即可;
2. 不用自己去創(chuàng)建文件對(duì)象和輸入、輸出流,提供文件名就可以返回File對(duì)象或輸入輸出流
3. 對(duì)于路徑操作返回的都是文件對(duì)象。
2.外部存儲(chǔ):
最容易混淆的是外部存儲(chǔ),如果說(shuō)pc上也要區(qū)分出外部存儲(chǔ)和內(nèi)部存儲(chǔ)的話,那么自帶的硬盤(pán)算是內(nèi)部存儲(chǔ),U盤(pán)或者移動(dòng)硬盤(pán)算是外部存儲(chǔ),因此我們很容易帶著這樣的理解去看待安卓手機(jī),認(rèn)為機(jī)身固有存儲(chǔ)是內(nèi)部存儲(chǔ),而擴(kuò)展的T卡是外部存儲(chǔ)。比如我們?nèi)蝿?wù)16GB版本的Nexus 4有16G的內(nèi)部存儲(chǔ),普通消費(fèi)者可以這樣理解,但是安卓的編程中不能,這16GB仍然是外部存儲(chǔ)。
所有的安卓設(shè)備都有外部存儲(chǔ)和內(nèi)部存儲(chǔ),這兩個(gè)名稱(chēng)來(lái)源于安卓的早期設(shè)備,那個(gè)時(shí)候的設(shè)備內(nèi)部存儲(chǔ)確實(shí)是固定的,而外部存儲(chǔ)確實(shí)是可以像U盤(pán)一樣移動(dòng)的。但是在后來(lái)的設(shè)備中,很多中高端機(jī)器都將自己的機(jī)身存儲(chǔ)擴(kuò)展到了8G以上,他們將存儲(chǔ)在概念上分成了"內(nèi)部internal" 和"外部external" 兩部分,但其實(shí)都在手機(jī)內(nèi)部。所以不管安卓手機(jī)是否有可移動(dòng)的sdcard,他們總是有外部存儲(chǔ)和內(nèi)部存儲(chǔ)。最關(guān)鍵的是,我們都是通過(guò)相同的api來(lái)訪問(wèn)可移動(dòng)的sdcard或者手機(jī)自帶的存儲(chǔ)(外部存儲(chǔ))。
外部存儲(chǔ)雖然概念上有點(diǎn)復(fù)雜,但也很好區(qū)分,你把手機(jī)連接電腦,能被電腦識(shí)別的部分就一定是外部存儲(chǔ)。
關(guān)于外部存儲(chǔ),我覺(jué)得api中在介紹Environment.getExternalStorageDirectory()方法的時(shí)候說(shuō)得很清楚:
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.
看不懂沒(méi)關(guān)系,其實(shí)跟我說(shuō)的意思差不多,只是覺(jué)得說(shuō)得比較形象,不知道是我的表述問(wèn)題,還是英文在邏輯解釋方面比漢語(yǔ)強(qiáng),因?yàn)榘自捨钠鋵?shí)是被閹割的漢語(yǔ)。
外部存儲(chǔ)中的文件是可以被用戶(hù)或者其他應(yīng)用程序修改的,有兩種類(lèi)型的文件(或者目錄):
1.公共文件Public files:文件是可以被自由訪問(wèn),且文件的數(shù)據(jù)對(duì)其他應(yīng)用或者用戶(hù)來(lái)說(shuō)都是由意義的,當(dāng)應(yīng)用被卸載之后,其卸載前創(chuàng)建的文件仍然保留。比如camera應(yīng)用,生成的照片大家都能訪問(wèn),而且camera不在了,照片仍然在。
如果你想在外存儲(chǔ)上放公共文件你可以使用getExternalStoragePublicDirectory()
1
2
3
4
5
6
7
8
9public 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;
}
在上面的代碼中我們創(chuàng)建獲得了存放picture的目錄,并且新創(chuàng)建一個(gè)albumName文件。
如果你的api 版本低于8,那么不能使用getExternalStoragePublicDirectory(),而是使用Environment.getExternalStorageDirectory(),他不帶參數(shù),也就不能自己創(chuàng)建一個(gè)目錄,只是返回外部存儲(chǔ)的根路徑。
2.私有文件Private files:其實(shí)由于是外部存儲(chǔ)的原因即是是這種類(lèi)型的文件也能被其他程序訪問(wèn),只不過(guò)一個(gè)應(yīng)用私有的文件對(duì)其他應(yīng)用其實(shí)是沒(méi)有訪問(wèn)價(jià)值的(惡意程序除外)。外部存儲(chǔ)上,應(yīng)用私有文件的價(jià)值在于卸載之后,這些文件也會(huì)被刪除。類(lèi)似于內(nèi)部存儲(chǔ)。
創(chuàng)建應(yīng)用私有文件的方法是Context.getExternalFilesDir(),如下:
1
2
3
4
5
6
7
8
9public 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;
}
上面的代碼創(chuàng)建了一個(gè)picture目錄,并在這個(gè)目錄下創(chuàng)建了一個(gè)名為albumName的文件,Environment.DIRECTORY_PICTURES其實(shí)就是字符串picture。
所有應(yīng)用程序的外部存儲(chǔ)的私有文件都放在根目錄的Android/data/下,目錄形式為/Android/data//
如果你的api 版本低于8,那么不能使用getExternalFilesDir(),而是使用Environment.getExternalStorageDirectory()獲得根路徑之后,自己再想辦法操作/Android/data//下的文件。
也就是說(shuō)api 8以下的版本在操作文件的時(shí)候沒(méi)有專(zhuān)門(mén)為私有文件和公共文件的操作提供api支持。你只能先獲取根目錄,然后自行想辦法。
在使用外部存儲(chǔ)之前,你必須要先檢查外部存儲(chǔ)的當(dāng)前狀態(tài),以判斷是否可用。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15boolean 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文件操作方法,我將這些方法的執(zhí)行結(jié)果打印出來(lái),看看到底路徑是啥樣,在activity中執(zhí)行以下代碼:
1
2
3
4
5
6Log.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中看到如下結(jié)果:
從log中我們可以看到外部存儲(chǔ)根目錄在我手機(jī)(nexus 3)上是/storage/emulated/0,奇怪的是在有些手機(jī)上同樣的代碼卻是下面的情況:
部存儲(chǔ)根目錄為/mnt/sdcard.
在網(wǎng)上搜了下好像是說(shuō)三星手機(jī)就是這樣。