摘要:字節(jié)流處理字節(jié)數(shù)據(jù)的流對象。寫入字符流的抽象類。是字符流通向字節(jié)流的橋梁可使用指定的將要寫入流中的字符編碼成字節(jié)。字節(jié)流是表示字節(jié)輸入流的所有類的超類。轉換流的最強功能就是基于字節(jié)流編碼表。刪除此抽象路徑名表示的文件或目錄。
IO流(重點理解)
用于處理設備上數(shù)據(jù)。
流:可以理解數(shù)據(jù)的流動,就是一個數(shù)據(jù)流。IO流最終要以對象來體現(xiàn),對象都存在IO包中。
流也進行分類:
1:輸入流(讀)和輸出流(寫)。
2:因為處理的數(shù)據(jù)不同,分為字節(jié)流和字符流。
字節(jié)流:處理字節(jié)數(shù)據(jù)的流對象。設備上的數(shù)據(jù)無論是圖片或者dvd,文字,它們都以二進制存儲的。二進制的最終都是以一個8位為數(shù)據(jù)單元進行體現(xiàn),所以計算機中的最小數(shù)據(jù)單元就是字節(jié)。意味著,字節(jié)流可以處理設備上的所有數(shù)據(jù),所以字節(jié)流一樣可以處理字符數(shù)據(jù)。
那么為什么要有字符流呢?因為字符每個國家都不一樣,所以涉及到了字符編碼問題,那么GBK編碼的中文用unicode編碼解析是有問題的,所以需要獲取中文字節(jié)數(shù)據(jù)的同時+ 指定的編碼表才可以解析正確數(shù)據(jù)。為了方便于文字的解析,所以將字節(jié)流和編碼表封裝成對象,這個對象就是字符流。只要操作字符數(shù)據(jù),優(yōu)先考慮使用字符流體系。
注意:流的操作只有兩種:讀和寫。
流的體系因為功能不同,但是有共性內容,不斷抽取,形成繼承體系。該體系一共有四個基類,而且都是抽象類。
字節(jié)流:InputStream OutputStream
字符流:Reader Writer
public static void main(String[] args) throws IOException { //讀、寫都會發(fā)生IO異常
/*
1:創(chuàng)建一個字符輸出流對象,用于操作文件。該對象一建立,就必須明確數(shù)據(jù)存儲位置,是一個文件。
2:對象產(chǎn)生后,會在堆內存中有一個實體,同時也調用了系統(tǒng)底層資源,在指定的位置創(chuàng)建了一個存儲數(shù)據(jù)的文件。
3:如果指定位置,出現(xiàn)了同名文件,文件會被覆蓋。
*/
FileWriter fw = new FileWriter("demo.txt"); // FileNotFoundException
/*
調用Writer類中的write方法寫入字符串。字符串并未直接寫入到目的地中,而是寫入到了流中,(其實是寫入到內存緩沖區(qū)中)。怎么把數(shù)據(jù)弄到文件中?
*/
fw.write("abcde");
fw.flush(); // 刷新緩沖區(qū),將緩沖區(qū)中的數(shù)據(jù)刷到目的地文件中。
fw.close(); // 關閉流,其實關閉的就是java調用的系統(tǒng)底層資源。在關閉前,會先刷新該流。
}
close()和flush()的區(qū)別:
flush():將緩沖區(qū)的數(shù)據(jù)刷到目的地中后,流可以使用。
io異常的處理方式:io一定要寫finally;
FileWriter寫入數(shù)據(jù)的細節(jié):
1:window中的換行符:rn兩個符號組成。 linux:n。
2:續(xù)寫數(shù)據(jù),只要在構造函數(shù)中傳入新的參數(shù)true。
3:目錄分割符:window /
public static void main(String[] args) {
FileWriter fw = null;
try {
fw = new FileWriter("demo.txt",true);
fw.write("abcde");
}
catch (IOException e ){
System.out.println(e.toString()+"....");
}
finally{
if(fw!=null)
try{
fw.close();
}
catch (IOException e){
System.out.println("close:"+e.toString());
}
}
FileReader:使用Reader體系,讀取一個文本文件中的數(shù)據(jù)。返回 -1 ,標志讀到結尾。
import java.io.*;
class FileReaderDemo {
public static void main(String[] args) throws IOException {
/*
創(chuàng)建可以讀取文本文件的流對象,F(xiàn)ileReader讓創(chuàng)建好的流對象和指定的文件相關聯(lián)。
*/
FileReader fr = new FileReader("demo.txt");
int ch = 0;
while((ch = fr.read())!= -1) { //條件是沒有讀到結尾
System.out.println((char)ch); //調用讀取流的read方法,讀取一個字符。
}
fr.close();
}
讀取數(shù)據(jù)的第二種方式:第二種方式較為高效,自定義緩沖區(qū)。
import java.io.*;
class FileReaderDemo2 {
public static void main(String[] args) throws IOException {
FileReader fr = new FileReader("demo.txt"); //創(chuàng)建讀取流對象和指定文件關聯(lián)。
//因為要使用read(char[])方法,將讀取到字符存入數(shù)組。所以要創(chuàng)建一個字符數(shù)組,一般數(shù)組的長度都是1024的整數(shù)倍。
char[] buf = new char[1024];
int len = 0;
while(( len=fr.read(buf)) != -1) {
System.out.println(new String(buf,0,len));
}
fr.close();
}
IO中的使用到了一個設計模式:裝飾設計模式。
裝飾設計模式解決:對一組類進行功能的增強。
包裝:寫一個類(包裝類)對被包裝對象進行包裝;
1、包裝類和被包裝對象要實現(xiàn)同樣的接口;
2、包裝類要持有一個被包裝對象;
3、包裝類在實現(xiàn)接口時,大部分方法是靠調用被包裝對象來實現(xiàn)的,對于需要修改的方法我們自己實現(xiàn);
字符流:
Reader:用于讀取字符流的抽象類。子類必須實現(xiàn)的方法只有 read(char[], int, int) 和 close()。
|---BufferedReader:從字符輸入流中讀取文本,緩沖各個字符,從而實現(xiàn)字符、數(shù)組和行的高效讀取。 可以指定緩沖區(qū)的大小,或者可使用默認的大小。大多數(shù)情況下,默認值就足夠大了。 |---LineNumberReader:跟蹤行號的緩沖字符輸入流。此類定義了方法 setLineNumber(int) 和 getLineNumber(),它們可分別用于設置和獲取當前行號。 |---InputStreamReader:是字節(jié)流通向字符流的橋梁:它使用指定的 charset 讀取字節(jié)并將其解碼為字符。它使用的字符集可以由名稱指定或顯式給定,或者可以接受平臺默認的字符集。 |---FileReader:用來讀取字符文件的便捷類。此類的構造方法假定默認字符編碼和默認字節(jié)緩沖區(qū)大小都是適當?shù)?。要自己指定這些值,可以先在 FileInputStream 上構造一個 InputStreamReader。 |---CharArrayReader: |---StringReader:
Writer:寫入字符流的抽象類。子類必須實現(xiàn)的方法僅有 write(char[], int, int)、flush() 和 close()。
|---BufferedWriter:將文本寫入字符輸出流,緩沖各個字符,從而提供單個字符、數(shù)組和字符串的高效寫入。 |---OutputStreamWriter:是字符流通向字節(jié)流的橋梁:可使用指定的 charset 將要寫入流中的字符編碼成字節(jié)。它使用的字符集可以由名稱指定或顯式給定,否則將接受平臺默認的字符集。 |---FileWriter:用來寫入字符文件的便捷類。此類的構造方法假定默認字符編碼和默認字節(jié)緩沖區(qū)大小都是可接受的。要自己指定這些值,可以先在 FileOutputStream 上構造一個 OutputStreamWriter。 |---PrintWriter: |---CharArrayWriter: |---StringWriter:
字節(jié)流:
InputStream:是表示字節(jié)輸入流的所有類的超類。
|--- FileInputStream:從文件系統(tǒng)中的某個文件中獲得輸入字節(jié)。哪些文件可用取決于主機環(huán)境。FileInputStream 用于讀取諸如圖像數(shù)據(jù)之類的原始字節(jié)流。要讀取字符流,請考慮使用 FileReader。 |--- FilterInputStream:包含其他一些輸入流,它將這些流用作其基本數(shù)據(jù)源,它可以直接傳輸數(shù)據(jù)或提供一些額外的功能。 |--- BufferedInputStream:該類實現(xiàn)緩沖的輸入流。 |--- Stream: |--- ObjectInputStream: |--- PipedInputStream:
OutputStream:此抽象類是表示輸出字節(jié)流的所有類的超類。
|--- FileOutputStream:文件輸出流是用于將數(shù)據(jù)寫入 File 或 FileDescriptor 的輸出流。 |--- FilterOutputStream:此類是過濾輸出流的所有類的超類。 |--- BufferedOutputStream:該類實現(xiàn)緩沖的輸出流。 |--- PrintStream: |--- DataOutputStream: |--- ObjectOutputStream: |--- PipedOutputStream:
緩沖區(qū)是提高效率用的,給誰提高呢?
BufferedWriter:是給字符輸出流提高效率用的,那就意味著,緩沖區(qū)對象建立時,必須要先有流對象。明確要提高具體的流對象的效率。
FileWriter fw = new FileWriter("bufdemo.txt");
BufferedWriter bufw = new BufferedWriter(fw);//讓緩沖區(qū)和指定流相關聯(lián)。
for(int x=0; x<4; x++){
bufw.write(x+"abc");
bufw.newLine(); //寫入一個換行符,這個換行符可以依據(jù)平臺的不同寫入不同的換行符。
bufw.flush();//對緩沖區(qū)進行刷新,可以讓數(shù)據(jù)到目的地中。
}
BufferedReader:
FileReader fr = new FileReader("bufdemo.txt");
BufferedReader bufr = new BufferedReader(fr);
String line = null;
while((line=bufr.readLine())!=null){ //readLine方法返回的時候是不帶換行符的。
System.out.println(line);
}
//記住,只要一讀取鍵盤錄入,就用這句話。
BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));
BufferedWriter bufw = new BufferedWriter(new OutputStreamWriter(System.out));//輸出到控制臺
String line = null;
while((line=bufr.readLine())!=null){
if("over".equals(line))
break;
bufw.write(line.toUpperCase());//將輸入的字符轉成大寫字符輸出
bufw.newLine();
bufw.flush();
}
bufw.close();
流對象:其實很簡單,就是讀取和寫入。但是因為功能的不同,流的體系中提供N多的對象。那么開始時,到底該用哪個對象更為合適呢?這就需要明確流的操作規(guī)律。
流的操作規(guī)律:
1,明確源和目的。
數(shù)據(jù)源:就是需要讀取,可以使用兩個體系:InputStream、Reader;
數(shù)據(jù)匯:就是需要寫入,可以使用兩個體系:OutputStream、Writer;
2,操作的數(shù)據(jù)是否是純文本數(shù)據(jù)?
如果是:數(shù)據(jù)源:Reader
數(shù)據(jù)匯:Writer
如果不是:數(shù)據(jù)源:InputStream
數(shù)據(jù)匯:OutputStream
3,雖然確定了一個體系,但是該體系中有太多的對象,到底用哪個呢?
明確操作的數(shù)據(jù)設備。
數(shù)據(jù)源對應的設備:硬盤(File),內存(數(shù)組),鍵盤(System.in)
數(shù)據(jù)匯對應的設備:硬盤(File),內存(數(shù)組),控制臺(System.out)。
4,需要在基本操作上附加其他功能嗎?比如緩沖。
如果需要就進行裝飾。
轉換流特有功能:轉換流可以將字節(jié)轉成字符,原因在于,將獲取到的字節(jié)通過查編碼表獲取到指定對應字符。
轉換流的最強功能就是基于 字節(jié)流 + 編碼表 。沒有轉換,沒有字符流。
發(fā)現(xiàn)轉換流有一個子類就是操作文件的字符流對象:
InputStreamReader
|--FileReader
OutputStreamWriter
|--FileWrier
想要操作文本文件,必須要進行編碼轉換,而編碼轉換動作轉換流都完成了。所以操作文件的流對象只要繼承自轉換流就可以讀取一個字符了。
但是子類有一個局限性,就是子類中使用的編碼是固定的,是本機默認的編碼表,對于簡體中文版的系統(tǒng)默認碼表是GBK。
FileReader fr = new FileReader("a.txt");
InputStreamReader isr = new InputStreamReader(new FileInputStream("a.txt"),"gbk");
以上兩句代碼功能一致,
如果僅僅使用平臺默認碼表,就使用FileReader fr = new FileReader("a.txt"); //因為簡化。
如果需要制定碼表,必須用轉換流。
轉換流 = 字節(jié)流+編碼表。
轉換流的子類File = 字節(jié)流 + 默認編碼表。
File類:將文件系統(tǒng)中的文件和文件夾封裝成了對象。提供了更多的屬性和行為可以對這些文件和文件夾進行操作。這些是流對象辦不到的,因為流只操作數(shù)據(jù)。
File類常見方法:
1:創(chuàng)建。
boolean createNewFile():在指定目錄下創(chuàng)建文件,如果該文件已存在,則不創(chuàng)建。而對操作文件的輸出流而言,輸出流對象已建立,就會創(chuàng)建文件,如果文件已存在,會覆蓋。除非續(xù)寫。
boolean mkdir():創(chuàng)建此抽象路徑名指定的目錄。
boolean mkdirs():創(chuàng)建多級目錄。
2:刪除。
boolean delete():刪除此抽象路徑名表示的文件或目錄。
void deleteOnExit():在虛擬機退出時刪除。
注意:在刪除文件夾時,必須保證這個文件夾中沒有任何內容,才可以將該文件夾用delete刪除。
window的刪除動作,是從里往外刪。注意:java刪除文件不走回收站。要慎用。
3:獲取.
long length():獲取文件大小。
String getName():返回由此抽象路徑名表示的文件或目錄的名稱。
String getPath():將此抽象路徑名轉換為一個路徑名字符串。
String getAbsolutePath():返回此抽象路徑名的絕對路徑名字符串。
String getParent():返回此抽象路徑名父目錄的抽象路徑名,如果此路徑名沒有指定父目錄,則返回 null。
long lastModified():返回此抽象路徑名表示的文件最后一次被修改的時間。
File.pathSeparator:返回當前系統(tǒng)默認的路徑分隔符,windows默認為 “;”。
File.Separator:返回當前系統(tǒng)默認的目錄分隔符,windows默認為 “”。
4:判斷:
boolean exists():判斷文件或者文件夾是否存在。
boolean isDirectory():測試此抽象路徑名表示的文件是否是一個目錄。
boolean isFile():測試此抽象路徑名表示的文件是否是一個標準文件。
boolean isHidden():測試此抽象路徑名指定的文件是否是一個隱藏文件。
boolean isAbsolute():測試此抽象路徑名是否為絕對路徑名。
5:重命名。
boolean renameTo(File dest):可以實現(xiàn)移動的效果。剪切+重命名。
String[] list():列出指定目錄下的當前的文件和文件夾的名稱。包含隱藏文件。
如果調用list方法的File 對象中封裝的是一個文件,那么list方法返回數(shù)組為null。如果封裝的對象不存在也會返回null。只有封裝的對象存在并且是文件夾時,這個方法才有效。遞歸:就是函數(shù)自身調用自身。
什么時候用遞歸呢?
當一個功能被重復使用,而每一次使用該功能時的參數(shù)不確定,都由上次的功能元素結果來確定。
簡單說:功能內部又用到該功能,但是傳遞的參數(shù)值不確定。(每次功能參與運算的未知內容不確定)。
遞歸的注意事項:
1:一定要定義遞歸的條件。
2:遞歸的次數(shù)不要過多。容易出現(xiàn) StackOverflowError 棧內存溢出錯誤。
Java.util.Properties:一個可以將鍵值進行持久化存儲的對象。Map--Hashtable的子類。
Map
|--Hashtable
|--Properties:用于屬性配置文件,鍵和值都是字符串類型。
特點:1:可以持久化存儲數(shù)據(jù)。2:鍵值都是字符串。3:一般用于配置文件。
|-- load():將流中的數(shù)據(jù)加載進集合。
原理:其實就是將讀取流和指定文件相關聯(lián),并讀取一行數(shù)據(jù),因為數(shù)據(jù)是規(guī)則的key=value,所以獲取一行后,通過 = 對該行數(shù)據(jù)進行切割,左邊就是鍵,右邊就是值,將鍵、值存儲到properties集合中。
|-- store():寫入各個項后,刷新輸出流。
以下介紹IO包中擴展功能的流對象:基本都是裝飾設計模式。
Java.io.outputstream.PrintStream:打印流
1:提供了更多的功能,比如打印方法??梢灾苯哟蛴∪我忸愋偷臄?shù)據(jù)。
2:它有一個自動刷新機制,創(chuàng)建該對象,指定參數(shù),對于指定方法可以自動刷新。
3:它使用的本機默認的字符編碼.
4:該流的print方法不拋出IOException。
該對象的構造函數(shù)。
PrintStream(File file) :創(chuàng)建具有指定文件且不帶自動行刷新的新打印流。
PrintStream(File file, String csn) :創(chuàng)建具有指定文件名稱和字符集且不帶自動行刷新的新打印流。
PrintStream(OutputStream out) :創(chuàng)建新的打印流。
PrintStream(OutputStream out, boolean autoFlush) :創(chuàng)建新的打印流。
PrintStream(OutputStream out, boolean autoFlush, String encoding) :創(chuàng)建新的打印流。
PrintStream(String fileName) :創(chuàng)建具有指定文件名稱且不帶自動行刷新的新打印流。
PrintStream(String fileName, String csn)
PrintStream可以操作目的:1:File對象。2:字符串路徑。3:字節(jié)輸出流。
前兩個都JDK1.5版本才出現(xiàn)。而且在操作文本文件時,可指定字符編碼了。
當目的是一個字節(jié)輸出流時,如果使用的println方法,可以在printStream對象上加入一個true參數(shù)。這樣對于println方法可以進行自動的刷新,而不是等待緩沖區(qū)滿了再刷新。最終print方法都將具體的數(shù)據(jù)轉成字符串,而且都對IO異常進行了內部處理。
既然操作的數(shù)據(jù)都轉成了字符串,那么使用PrintWriter更好一些。因為PrintWrite是字符流的子類,可以直接操作字符數(shù)據(jù),同時也可以指定具體的編碼。PrintWriter:具備了PrintStream的特點同時,還有自身特點:
該對象的目的地有四個:1:File對象。2:字符串路徑。3:字節(jié)輸出流。4:字符輸出流。
開發(fā)時盡量使用PrintWriter。
方法中直接操作文件的第二參數(shù)是編碼表。
直接操作輸出流的,第二參數(shù)是自動刷新。
//讀取鍵盤錄入將數(shù)據(jù)轉成大寫顯示在控制臺.
BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));//源:鍵盤輸入
//目的:把數(shù)據(jù)寫到文件中,還想自動刷新。
PrintWriter out = new PrintWriter(new FileWriter("out.txt"),true);//設置true后自動刷新
String line = null;
while((line=bufr.readLine())!=null){
if("over".equals(line))
break;
out.println(line.toUpperCase());//轉大寫輸出
}
//注意:System.in,System.out這兩個標準的輸入輸出流,在jvm啟動時已經(jīng)存在了。隨時可以使用。當jvm結束了,這兩個流就結束了。但是,當使用了顯示的close方法關閉時,這兩個流在提前結束了。
out.close();
SequenceInputStream:序列流,作用就是將多個讀取流合并成一個讀取流。實現(xiàn)數(shù)據(jù)合并。
表示其他輸入流的邏輯串聯(lián)。它從輸入流的有序集合開始,并從第一個輸入流開始讀取,直到到達文件末尾,接著從第二個輸入流讀取,依次類推,直到到達包含的最后一個輸入流的文件末尾為止。
這樣做,可以更方便的操作多個讀取流,其實這個序列流內部會有一個有序的集合容器,用于存儲多個讀取流對象。
該對象的構造函數(shù)參數(shù)是枚舉,想要獲取枚舉,需要有Vector集合,但不高效。需用ArrayList,但ArrayList中沒有枚舉,只有自己去創(chuàng)建枚舉對象。
但是方法怎么實現(xiàn)呢?因為枚舉操作的是具體集合中的元素,所以無法具體實現(xiàn),但是枚舉和迭代器是功能一樣的,所以,可以用迭代替代枚舉。
合并原理:多個讀取流對應一個輸出流。
切割原理:一個讀取流對應多個輸出流。
import java.io.*;
import java.util.*;
class SplitFileDemo{
private static final String CFG = ".properties";
private static final String SP = ".part";
public static void main(String[] args) throws IOException{
File file = new File("c: