1. 程式人生 > >Java中級進階之IO流

Java中級進階之IO流

IO流

1.流

概念:流就是一系列的資料

1.1 什麼是流

① 當不同的介質之間有資料互動的時候,JAVA就使用流來實現

② 資料來源可以是檔案,還可以是資料庫,網路甚至是其他的程式

③ 比如讀取檔案的資料到程式中,站在程式的角度來看,就叫做輸入流

輸入流: InputStream

輸出流:OutputStream

1.2 檔案輸入流

① 建立了一個檔案輸入流,這個流可以用來把資料從硬碟的檔案,讀取到JVM(記憶體)。

程式碼:

1File f = new File("d:/test.txt");
2// 建立基於檔案的輸入流
3FileInputStream fis = new FileInputStream(f);

2. 位元組流

概念:用於以位元組的形式讀取和寫入資料『InputStream、OutputStream』

2.1 ASCII碼 概念

① 所有的資料存放在計算機中都是以數字的形式存放的。 所以字母就需要轉換為數字才能夠存放。

② 比如:A就對應的數字65,a對應的數字97. 不同的字母和符號對應不同的數字,就是一張碼錶。

③ ASCII是這樣的一種碼錶。 只包含簡單的英文字母、符號、數字等等。

2.2 以位元組流的形式讀取檔案內容

① InputStream是位元組輸入流,同時也是抽象類,只提供方法宣告,不提供方法的具體實現。

② FileInputStream 是InputStream子類,以FileInputStream 為例進行檔案讀取

程式碼:

 1//準備檔案test.txt 其中的內容是AB,對應的ASCII分別是65 66
 2File f =new File("d:/test.txt");
 3FileInputStream fis =new FileInputStream(f);
 4//建立位元組陣列,其長度就是檔案的長度
 5byte[] all =new byte[(int) f.length()];
 6//以位元組流的形式讀取檔案所有內容
 7fis.read(all);
 8for (byte b : all) {
 9//打印出來是65 66
10System.out.println(b);
11}
12//每次使用完流,都應該進行關閉
13fis.close();

2.3 以位元組流的形式讀取檔案內容

① OutputStream是位元組輸出流,同時也是抽象類,只提供方法宣告,不提供方法的具體實現。

② FileOutputStream 是OutputStream子類,以FileOutputStream 為例向檔案寫出資料。

注: 如果檔案d:/test2.txt不存在,寫出操作會自動建立該檔案。但是如果是檔案 d:/xyz/test2.txt,而目錄xyz又不存在,會丟擲異常

程式碼:

 1// 準備檔案test2.txt其中的內容是空的
 2File f = new File("d:/test2.txt");
 3// 準備長度是2的位元組陣列,用88,89初始化,其對應的字元分別是X,Y
 4byte data[] = { 88, 89 };
 5// 建立基於檔案的輸出流
 6FileOutputStream fos = new FileOutputStream(f);
 7// 把資料寫入到輸出流
 8fos.write(data);
 9// 關閉輸出流
10fos.close();
11} catch (IOException e) {
12// TODO Auto-generated catch block
13e.printStackTrace();
14}

3. 字元流

概念:Reader字元輸入流、Writer字元輸出流、專門用於字元的形式讀取和寫入資料。

3.1使用字元流讀取檔案

程式碼:

 1// 準備檔案test.txt其中的內容是AB
 2File f = new File("d:/test.txt");
 3// 建立基於檔案的Reader
 4try (FileReader fr = new FileReader(f)) {
 5// 建立字元陣列,其長度就是檔案的長度
 6char[] all = new char[(int) f.length()];
 7// 以字元流的形式讀取檔案所有內容
 8fr.read(all);
 9for (char b : all) {
10// 打印出來是A B
11System.out.println(b);
12}
13} catch (IOException e) {
14// TODO Auto-generated catch block
15e.printStackTrace();
16}

3.2 使用字元流把字串寫入到檔案

程式碼:

 1// 準備檔案test.txt
 2File f = new File("d:/test.txt");
 3// 建立基於檔案的Writer
 4try (FileWriter fr = new FileWriter(f)) {
 5// 以字元流的形式把資料寫入到檔案中
 6String data="abcdefg";
 7char[] cs = data.toCharArray();
 8fr.write(cs);
 9} catch (IOException e) {
10// TODO Auto-generated catch block
11e.printStackTrace();
12}

4.快取流

介紹:以介質是硬碟為例,位元組流和字元流的弊端:

① 在每一次讀寫的時候,都會訪問硬碟。 如果讀寫的頻率比較高的時候,其效能表現不佳。

② 為了解決以上弊端,採用快取流。

③ 快取流在讀取的時候,會一次性讀較多的資料到快取中,以後每一次的讀取,都是在快取中訪問,直到快取中的資料讀取完畢,再到硬碟中讀取。

④ 就好比吃飯,不用快取就是每吃一口都到鍋裡去鏟。用快取就是先把飯盛到碗裡,碗裡的吃完了,再到鍋裡去鏟

⑤ 快取流在寫入資料的時候,會先把資料寫入到快取區,直到快取區達到一定的量,才把這些資料,一起寫入到硬碟中去。按照這種操作模式,就不會像位元組流,字元流那樣每寫一個位元組都訪問硬碟,從而減少了IO操作

4.1 使用快取流讀取資料

程式碼:

 1public static void main(String[] args) {
 2// 準備檔案test.txt其中的內容是
 3// sdasd
 4//asdasdasd
 5// sadsfadc
 6File f = new File("d:/test.txt");
 7// 建立檔案字元流
 8// 快取流必須建立在一個存在的流的基礎上
 9try (
10FileReader fr = new FileReader(f);
11BufferedReader br = new BufferedReader(fr);
12)
13{
14while (true) {
15// 一次讀一行
16String line = br.readLine();
17if (null == line)
18break;
19System.out.println(line);
20}
21} catch (IOException e) {
22// TODO Auto-generated catch block
23e.printStackTrace();
24}

4.2 使用快取流寫出資料

概念:PrintWriter 快取字元輸出流, 可以一次寫出一行資料

程式碼:

 1// 向檔案test2.txt中寫入三行語句
 2File f = new File("d:/test2.txt");
 3try (
 4// 建立檔案字元流
 5FileWriter fw = new FileWriter(f);
 6// 快取流必須建立在一個存在的流的基礎上
 7PrintWriter pw = new PrintWriter(fw);
 8) {
 9pw.println("garen kill teemo");
10pw.println("teemo revive after 1 minutes");
11pw.println("teemo try to garen, but killed again");
12} catch (IOException e) {
13// TODO Auto-generated catch block
14e.printStackTrace();
15}

4.3 flush

概念:有的時候,需要立即把資料寫入到硬碟,而不是等快取滿了才寫出去。 這時候就需要用到flush

程式碼:

 1//向檔案test2.txt中寫入三行語句
 2File f =new File("d:/test2.txt");
 3//建立檔案字元流
 4//快取流必須建立在一個存在的流的基礎上
 5try(FileWriter fr = new FileWriter(f);PrintWriter pw = newPrintWriter(fr);) {
 6pw.println("garen kill teemo");
 7//強制把快取中的資料寫入硬碟,無論快取是否已滿
 8pw.flush();
 9pw.println("teemo revive after 1 minutes");
10pw.flush();
11pw.println("teemo try to garen, but killed again");
12pw.flush();
13} catch (IOException e) {
14// TODO Auto-generated catch block
15e.printStackTrace();
16}
17}

5. 資料流

介紹:

① 直接進行字串的讀寫

② 使用資料流的writeUTF()和readUTF() 可以進行資料的格式化順序讀寫

③ 如本例,通過DataOutputStream 向檔案順序寫出 布林值,整數和字串。 然後再通過DataInputStream 順序讀入這些資料。

④ 注: 要用DataInputStream 讀取一個檔案,這個檔案必須是由DataOutputStream 寫出的,否則會出現EOFException,因為DataOutputStream 在寫出的時候會做一些特殊標記,只有DataInputStream 才能成功的讀取。

程式碼:

 1public static void main(String[] args) {
 2write();
 3read();
 4}
 5private static void read() {
 6File f =new File("d:/test.txt");
 7try (
 8FileInputStream fis  = new FileInputStream(f);
 9DataInputStream dis =new DataInputStream(fis);
10){
11boolean b= dis.readBoolean();
12int i = dis.readInt();
13String str = dis.readUTF();
14System.out.println("讀取到布林值:"+b);
15System.out.println("讀取到整數:"+i);
16System.out.println("讀取到字串:"+str);
17} catch (IOException e) {
18e.printStackTrace();
19}
20}
21private static void write() {
22File f =new File("d:/test.txt");
23try (
24FileOutputStream fos  = newFileOutputStream(f);
25DataOutputStream dos =newDataOutputStream(fos);
26){
27dos.writeBoolean(false);
28dos.writeInt(200);
29dos.writeUTF("123 wew d wd s ");
30} catch (IOException e) {
31e.printStackTrace();
32}
33}

6. 物件流

6.1 介紹

① 物件流指的是可以直接把一個物件以流的形式傳輸給其他的介質,比如硬碟

② 一個物件以流的形式進行傳輸,叫做序列化。 該物件所對應的類,必須是實現Serializable介面

6.2序列化一個物件

①建立一個H物件,設定其名稱為g。

②把該物件序列化到一個檔案g.txt。

然後再通過序列化把該檔案轉換為一個H物件。

注:把一個物件序列化有一個前提是:這個物件的類,必須實現了Serializable介面。

程式碼:

 1public class Hero implements Serializable {
 2//表示這個類當前的版本,如果有了變化,比如新設計了屬性,就應該修改這個版本號
 3private static final long serialVersionUID = 1L;
 4public String name;
 5public float hp;
 6}
 7public class TestStream {
 8public static void main(String[] args) {
 9//建立一個Hero garen
10//要把Hero物件直接儲存在檔案上,務必讓Hero類實現Serializable介面
11Hero h = new Hero();
12h.name = "garen";
13h.hp = 616;
14//準備一個檔案用於儲存該物件
15File f =new File("d:/garen.lol");
16try(
17//建立物件輸出流
18FileOutputStream fos = new FileOutputStream(f);
19ObjectOutputStream oos =new ObjectOutputStream(fos);
20//建立物件輸入流
21FileInputStream fis = new FileInputStream(f);
22ObjectInputStream ois =new ObjectInputStream(fis);
23) {
24oos.writeObject(h);
25Hero h2 = (Hero) ois.readObject();
26System.out.println(h2.name);
27System.out.println(h2.hp);
28} catch (IOException e) {
29// TODO Auto-generated catch block
30e.printStackTrace();
31} catch (ClassNotFoundException e) {
32// TODO Auto-generated catch block
33e.printStackTrace();
34}
35}
36}

2. 關閉流的方式

概念:所有的流,無論是輸入流還是輸出流,使用完畢之後,都應該關閉。 如果不關閉,會產生對資源佔用的浪費。 當量比較大的時候,會影響到業務的正常開展。

2.1 在try中關閉

弊端:如果檔案不存在,或者讀取的時候出現問題而丟擲異常,那麼就不會執行這一行關閉流的程式碼,存在巨大的資源佔用隱患。

程式碼:

 1try {
 2File f = new File("d:/test.txt");
 3FileInputStream fis = new FileInputStream(f);
 4byte[] all = new byte[(int) f.length()];
 5fis.read(all);
 6for (byte b : all) {
 7System.out.println(b);
 8}
 9// 在try 裡關閉流
10fis.close();
11} catch (IOException e) {
12e.printStackTrace();
13}

2.2  在finally中關閉「標準的關閉流的方式」

介紹:

①  首先把流的引用宣告在 try 的外面,如果宣告在 try 裡面,其作用域無法抵達finally .

②  在 finally 關閉之前,要先判斷該引用是否為空

③ 關閉的時候,需要再一次進行 try catch 處理

程式碼:

 1File f = new File("d:/test.txt");
 2FileInputStream fis = null;
 3try {
 4fis = new FileInputStream(f);
 5byte[] all = new byte[(int) f.length()];
 6fis.read(all);
 7for (byte b : all) {
 8System.out.println(b);
 9}
10} catch (IOException e) {
11e.printStackTrace();
12} finally {
13// 在finally 裡關閉流
14if (null != fis)
15try {
16fis.close();
17} catch (IOException e) {
18// TODO Auto-generated catch block
19e.printStackTrace();
20}
21}

2.3 使用 try() 的方式「把流定義在 try() 裡,try 、catch 或者 finally 結束的時候,會自動關閉」

程式碼:

 1File f = new File("d:/test.txt");
 2//把流定義在try()裡,try,catch或者finally結束的時候,會自動關閉
 3try (FileInputStream fis = new FileInputStream(f)) {
 4byte[] all = new byte[(int) f.length()];
 5fis.read(all);
 6for (byte b : all) {
 7System.out.println(b);
 8}
 9} catch (IOException e) {
10e.printStackTrace();
11}