先來看一下流的概念:java
在程序中全部的數據都是以流的方式進行傳輸或保存的,程序須要數據的時候要使用輸入流讀取數據,而當程序須要將一些數據保存起來的時候,就要使用輸出流完成。數組
程序中的輸入輸出都是以流的形式保存的,流中保存的實際上全都是字節文件。app
字節流與字符流函數
在java.io包中操做文件內容的主要有兩大類:字節流、字符流,兩類都分爲輸入和輸出操做。在字節流中輸出數據主要是使用OutputStream完成,輸入使的是InputStream,在字符流中輸出主要是使用Writer類完成,輸入流主要使用Reader類完成。(這四個都是抽象類)編碼
java中提供了專用於輸入輸出功能的包Java.io,其中包括:
InputStream,OutputStream,Reader,Writer
InputStream 和OutputStream,兩個是爲字節流設計的,主要用來處理字節或二進制對象,
Reader和 Writer.兩個是爲字符流(一個字符佔兩個字節)設計的,主要用來處理字符或字符串.spa
字符流處理的單元爲2個字節的Unicode字符,分別操做字符、字符數組或字符串,而字節流處理單元爲1個字節,操做字節和字節數組。因此字符流是由Java虛擬機將字節轉化爲2個字節的Unicode字符爲單位的字符而成的,因此它對多國語言支持性比較好!若是是音頻文件、圖片、歌曲,就用字節流好點,若是是關係到中文(文本)的,用字符流好點
全部文件的儲存是都是字節(byte)的儲存,在磁盤上保留的並非文件的字符而是先把字符編碼成字節,再儲存這些字節到磁盤。在讀取文件(特別是文本文件)時,也是一個字節一個字節地讀取以造成字節序列設計
字節流可用於任何類型的對象,包括二進制對象,而字符流只能處理字符或者字符串; 字節流提供了處理任何類型的IO操做的功能,但它不能直接處理Unicode字符,而字符流就能夠
字節流是最基本的,全部的InputStrem和OutputStream的子類都是,主要用在處理二進制數據,它是按字節來處理的 但實際中不少的數據是文本,又提出了字符流的概念,它是按虛擬機的encode來處理,也就是要進行字符集的轉化 這兩個之間經過 InputStreamReader,OutputStreamWriter來關聯,其實是經過byte[]和String來關聯 在實際開發中出現的漢字問題實際上都是在字符流和字節流之間轉化不統一而形成的 。code
字節流和字符流的操做過程: 對象
以文件操做爲例,主要的操做流程以下:blog
1 使用File類打開一個文件
2 經過字節流或字符流的子類,指定輸出的位置
3 進行讀/寫操做
4 關閉輸入/輸出
IO操做屬於資源操做,必定要記得關閉
字節流的操做:
字節流主要是操做byte類型數據,以byte數組爲準,主要操做類就是OutputStream、InputStream
字節輸出流:OutputStream
OutputStream是整個IO包中字節輸出流的最大父類,此類的定義以下:
public abstract class OutputStream extends Object implements Closeable,Flushable
從以上的定義能夠發現,此類是一個抽象類,若是想要使用此類的話,則首先必須經過子類實例化對象,那麼若是如今要操做的是一個文件,則可使用:FileOutputStream類。經過向上轉型以後,能夠爲OutputStream實例化
Closeable表示能夠關閉的操做,由於程序運行到最後確定要關閉
Flushable:表示刷新,清空內存中的數據
FileOutputStream類的構造方法以下:
public FileOutputStream(File file)throws FileNotFoundException
寫數據:
import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.OutputStream; public class Test11 { public static void main(String[] args) throws IOException { File f = new File("d:" + File.separator+"test.txt"); OutputStream out=new FileOutputStream(f);//若是文件不存在會自動建立 String str="Hello World"; byte[] b=str.getBytes(); out.write(b);//由於是字節流,因此要轉化成字節數組進行輸出 out.close(); } } 也能夠一個字節一個字節進行輸出,以下: import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.OutputStream; public class Test11 { public static void main(String[] args) throws IOException { File f = new File("d:" + File.separator+"test.txt"); OutputStream out=new FileOutputStream(f);//若是文件不存在會自動建立 String str="Hello World"; byte[] b=str.getBytes(); for(int i=0;i<b.length;i++){ out.write(b[i]); } out.close(); } }
以上輸出只會進行覆蓋,若是要追加的話,請看FileOutputStream類的另外一個構造方法:
public FileOutputStream(File file,boolean append)throws FileNotFoundException
在構造方法中,若是將append的值設置爲true,則表示在文件的末尾追加內容。
import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.OutputStream; public class Test11 { public static void main(String[] args) throws IOException { File f = new File("d:" + File.separator+"test.txt"); OutputStream out=new FileOutputStream(f,true);//追加內容 String str="\r\nHello World"; byte[] b=str.getBytes(); for(int i=0;i<b.length;i++){ out.write(b[i]); } out.close(); } }
字節輸入流:InputStream
既然程序能夠向文件中寫入內容,則就能夠經過InputStream從文件中把內容讀取進來,首先來看InputStream類的定義:
public abstract class InputStream extends Object implements Closeable
與OutputStream類同樣,InputStream自己也是一個抽象類,必須依靠其子類,若是如今是從文件中讀取,就用FileInputStream來實現。
觀察FileInputStream類的構造方法:
public FileInputStream(File file)throws FileNotFoundException
讀文件:
import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; public class Test12 { public static void main(String[] args) throws IOException { File f = new File("d:" + File.separator+"test.txt"); InputStream in=new FileInputStream(f); byte[] b=new byte[(int) f.length()]; //根據文件的大小來定義字節數組的大小int len=in.read(b); in.close(); System.out.println(new String(b,0,len)); } }
換種方式,一個字節一個字節讀入
import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; public class Test14 { public static void main(String[] args) throws IOException { File f = new File("d:" + File.separator+"test.txt"); InputStream in=new FileInputStream(f); byte[] b=new byte[(int) f.length()]; for(int i=0;i<b.length;i++){ b[i]=(byte) in.read(); } in.close(); System.out.println(new String(b)); } }
但以上狀況只適合知道輸入文件的大小,不知道的話用以下方法:
import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; public class Test15 { public static void main(String[] args) throws IOException { File f = new File("d:" + File.separator+"test.txt"); InputStream in=new FileInputStream(f); byte[] b=new byte[1024]; int temp=0; int len=0; while((temp=in.read())!=-1){//-1爲文件讀完的標誌 b[len]=(byte) temp; len++; } in.close(); System.out.println(new String(b,0,len)); } }
字符流
在程序中一個字符等於兩個字節,那麼java提供了Reader、Writer兩個專門操做字符流的類。
字符輸出流:Writer
Writer自己是一個字符流的輸出類,此類的定義以下:
public abstract class Writer extends Object implements Appendable,Closeable,Flushable
此類自己也是一個抽象類,若是要使用此類,則確定要使用其子類,此時若是是向文件中寫入內容,因此應該使用FileWriter的子類。
FileWriter類的構造方法定義以下:
public FileWriter(File file)throws IOException
字符流的操做比字節流操做好在一點,就是能夠直接輸出字符串了,不用再像以前那樣進行轉換操做了。
寫文件:
import java.io.File; import java.io.FileWriter; import java.io.IOException; import java.io.Writer; public class Test16 { public static void main(String[] args) throws IOException { File f = new File("d:" + File.separator+"test.txt"); Writer out=new FileWriter(f); String str="Hello World"; out.write(str); out.close(); } } 在默認狀況下再次輸出會覆蓋,追加的方法也是在構造函數上加上追加標記 import java.io.File; import java.io.FileWriter; import java.io.IOException; import java.io.Writer; public class Test17 { public static void main(String[] args) throws IOException { File f = new File("d:" + File.separator+"test.txt"); Writer out=new FileWriter(f,true);//追加 String str="\r\nHello World"; out.write(str); out.close(); } }
字符輸入流:Reader
Reader是使用字符的方式從文件中取出數據,Reader類的定義以下:
public abstract class Reader extends Objects implements Readable,Closeable
Reader自己也是抽象類,若是如今要從文件中讀取內容,則能夠直接使用FileReader子類。
FileReader的構造方法定義以下:
public FileReader(File file)throws FileNotFoundException
以字符數組的形式讀取出數據:
import java.io.File; import java.io.FileReader; import java.io.IOException; import java.io.Reader; public class Test18 { public static void main(String[] args) throws IOException { File f = new File("d:" + File.separator+"test.txt"); Reader input=new FileReader(f); char[] c=new char[1024]; int len=input.read(c); input.close(); System.out.println(new String(c,0,len)); } } 也能夠用循環方式,判斷是否讀到底 import java.io.File; import java.io.FileReader; import java.io.IOException; import java.io.Reader; public class Test19 { public static void main(String[] args) throws IOException { File f = new File("d:" + File.separator+"test.txt"); Reader input=new FileReader(f); char[] c=new char[1024]; int temp=0; int len=0; while((temp=input.read())!=-1){ c[len]=(char) temp; len++; } input.close(); System.out.println(new String(c,0,len)); } }
字節流與字符流的區別
字節流和字符流使用是很是類似的,那麼除了操做代碼的不一樣以外,還有哪些不一樣呢?
字節流在操做的時候自己是不會用到緩衝區(內存)的,是與文件自己直接操做的,而字符流在操做的時候是使用到緩衝區的
字節流在操做文件時,即便不關閉資源(close方法),文件也能輸出,可是若是字符流不使用close方法的話,則不會輸出任何內容,說明字符流用的是緩衝區,而且可使用flush方法強制進行刷新緩衝區,這時才能在不close的狀況下輸出內容
那開發中究竟用字節流好仍是用字符流好呢?
在全部的硬盤上保存文件或進行傳輸的時候都是以字節的方法進行的,包括圖片也是按字節完成,而字符是隻有在內存中才會造成的,因此使用字節的操做是最多的。
若是要java程序實現一個拷貝功能,應該選用字節流進行操做(可能拷貝的是圖片),而且採用邊讀邊寫的方式(節省內存)。
雖然Java支持字節流和字符流,但有時須要在字節流和字符流二者之間轉換。InputStreamReader和OutputStreamWriter,這兩個爲類是字節流和字符流之間相互轉換的類。
InputSreamReader用於將一個字節流中的字節解碼成字符:
有兩個構造方法:
InputStreamReader(InputStream in);
功能:用默認字符集建立一個InputStreamReader對象
InputStreamReader(InputStream in,String CharsetName);
功能:接收已指定字符集名的字符串,並用該字符建立對象
OutputStream用於將寫入的字符編碼成字節後寫入一個字節流。
一樣有兩個構造方法:
OutputStreamWriter(OutputStream out);
功能:用默認字符集建立一個OutputStreamWriter對象;
OutputStreamWriter(OutputStream out,String CharSetName);
功能:接收已指定字符集名的字符串,並用該字符集建立OutputStreamWrite對象
爲了不頻繁的轉換字節流和字符流,對以上兩個類進行了封裝。
BufferedWriter類封裝了OutputStreamWriter類;
BufferedReader類封裝了InputStreamReader類;
封裝格式:
BufferedWriter out=new BufferedWriter(new OutputStreamWriter(System.out));
BufferedReader in= new BufferedReader(new InputStreamReader(System.in);
利用下面的語句,能夠從控制檯讀取一行字符串:
BufferedReader in=new BufferedReader(new InputStreamReader(System.in));
String line=in.readLine();