1. 程式人生 > 其它 >Linux進階內容

Linux進階內容

為什麼會出現字元流

由於位元組流操作中文不是特別的方便,所以Java就提供字元流

  • 字元流 = 位元組流 + 編碼表

用字元流複製文字檔案時,文字檔案也會有中文,但是沒有問題,原因是最終底層操作會自動進行位元組拼接成中文,如何識別是中文呢?

  • 漢字在儲存的時候,無論選擇哪種編碼儲存,第一個位元組都是負數

 

 1 import java.io.FileInputStream;
 2 import java.io.IOException;
 3 import java.util.Arrays;
 4 
 5 /*
 6 需求:位元組流讀文字檔案資料
 7 
 8     一個漢字的儲存:
 9
如果是GBK編碼,佔用2個位元組 10 如果是UTF-8編碼,佔用3個位元組 11 12 */ 13 14 public class FileInputStreamDemo3 { 15 public static void main(String[] args) throws IOException{ 16 // FileInputStream fis=new FileInputStream("..\\hello java\\a.txt"); 17 // 18 // int by; 19 // while((by=fis.read())!=-1) {
20 // System.out.print((char)by); 21 // } 22 // 23 // fis.close(); 24 25 // String s="abc";//[97, 98, 99] 26 27 String s="中國";//[-42, -48, -71, -6] 預設GBK編碼 28 // byte[] bys = s.getBytes(); 29 30 // byte[] bys = s.getBytes("UTF-8");
//[-28, -72, -83, -27, -101, -67] 31 byte[] bys = s.getBytes("GBK");//[-42, -48, -71, -6] 32 System.out.println(Arrays.toString(bys)); 33 34 35 } 36 }

編碼表

基礎知識:

    • 計算機中儲存的資訊都是用二進位制數表示的;我們在螢幕上看到的英文、漢字等字元是二進位制數轉換之後的結果
    • 按照某種規則,將字元儲存到計算機中,稱為編碼。反之,將儲存在計算機中的二進位制數按照某種規則解析顯示出來,稱為解碼。這裡強調一下:按照A編碼儲存,必須按照A編碼解析,這樣才能顯示正確的文字符號。否則就會導致亂碼現象。

字元編碼:就是一套自然語言的字元與二進位制數之間的對應規則(A,65)


字符集:

    • 是一個系統支援的所有字元的集合,包括各國家文字、標點符號、圖形符號、數字等
    • 計算機要準確的儲存和識別個只能怪字符集符號,就需要進行字元編碼,一套字符集必然至少有一套字元編碼。

常見字符集有ASCII字符集、GBXXX字符集、Unicode字符集等

ASCII字符集:

    • ASCII(American Standard Code for Information Interchange,美國資訊交換標準程式碼):是基於拉丁字母的一套電腦編碼系統,用於顯示現代英文,主要包括控制字元(回車鍵、退格、換行鍵等)和可顯示字元(英文大小寫字元、阿拉伯數字和西文符號)
    • 基本的ASCII字符集,使用7為表示一個字元,共128字元。ASCII的擴充套件字符集使用8位表示一個字元,共256字元,方便支援歐洲常用字元。是一個系統支援的所有字元的集合,包括各國家文字、標點符號、圖形符號、數字等

GBXXX字符集:

    • GB2312:簡體中文碼錶。一個小於127的字元的意義與原來相同,但兩個大於127的字元連在一起時,就表示一個漢字,這樣大約可以組合了包含7000多個簡體漢字,此外數學符號、羅馬希臘的字母、日文的假名等都編進去了,連在ASCII裡本來就有的數字、標點、字母都統統重寫編了兩個位元組長的編碼,這就是常說的“全形”字元,而原來在127號以下的那些就餓叫“半形”字元了
    • GBK: 最常用的中文碼錶。是在GB2312標準基礎上擴充套件規範,使用了雙位元組編碼方案,共收錄了21003個漢字,完全相容GB2312標準,同時支援繁體漢字以及日韓漢字等
    • GB18030:最新的中文碼錶。共收錄漢字70244個,採用多位元組編碼,每個字可以有1個、2個或4個位元組組成。支援中國國內少數民族的文字,同時支援繁體漢字以及日韓漢字等

Unicode字符集:

    • 為表達任意語言的任意字元而設計,是業界的一種標準,也稱為統一碼、標準萬國碼。它最多使用4個位元組的數字來表達每個字母、符號、或者文字。有三種編碼方案,UTF-8、UTF-16和UTF32。最為常用的UTF-8編碼
    • UTF-8編碼:可以用來表示Unicode標準中任意字元,它是電子郵件、網頁及其他儲存或傳送文字的應用中,優先採用的編碼。網際網路工程工作小組(IETF)要求所有網際網路協議都必須支援UTF-8編碼。它使用一至4個位元組為每個字元編碼

編碼規則:
128個US-ASCII字元,只需一個位元組編碼
拉丁文等文字,需要二個位元組編碼
大部分常用字(含中文),使用三個位元組編碼
其他極少使用的Unicode輔助字元,使用四位元組編碼

 

小結:採用何種規則編碼,就要採用對應規則解碼,否則就會出現亂碼

字串中的編碼解碼

編碼:

    • byte[] getBytes():使用平臺的預設字符集將該String編碼為以一系列位元組,將結果儲存到新的位元組陣列中
    • byte[] getBytes(String charsetName):使用指定的字符集將該String編碼為一系列位元組,將結果儲存到新的位元組陣列中

解碼:

    • String(byte[] bytes):通過使用平臺的預設字符集解碼指定的位元組陣列來構造新的String
    • String(byte[] bytes,String charsetName):通過指定的字符集解碼指定的位元組陣列來構造新的String
 1 import java.io.UnsupportedEncodingException;
 2 import java.util.Arrays;
 3 
 4 /*
 5 編碼:
 6 
 7 - byte[] getBytes():使用平臺的預設字符集將該String編碼為以一系列位元組,將結果儲存到新的位元組陣列中
 8 - byte[] getBytes(String charsetName):使用指定的字符集將該String編碼為一系列位元組,將結果儲存到新的位元組陣列中
 9 
10 解碼:
11 
12 - String(byte[] bytes):通過使用平臺的預設字符集解碼指定的位元組陣列來構造新的String
13 - String(byte[] bytes,String charsetName):通過指定的字符集解碼指定的位元組陣列來構造新的String
14 
15 */
16 public class StringDemo {
17     public static void main(String[] args) throws UnsupportedEncodingException {
18         //定義一個字串
19         String s="中國";
20         //編碼:
21         //byte[] getBytes():使用平臺的預設字符集將該String編碼為以一系列位元組,將結果儲存到新的位元組陣列中
22         byte[] bys = s.getBytes();//[-42, -48, -71, -6]
23         
24         //byte[] getBytes(String charsetName):使用指定的字符集將該String編碼為一系列位元組,將結果儲存到新的位元組陣列中
25 //        byte[] bys = s.getBytes("UTF-8");//[-28, -72, -83, -27, -101, -67]
26 //        byte[] bys = s.getBytes("GBK");//[-42, -48, -71, -6]
27         System.out.println(Arrays.toString(bys));
28         
29         //解碼:
30         //String(byte[] bytes):通過使用平臺的預設字符集解碼指定的位元組陣列來構造新的String
31 //        String ss=new String(bys);//中國
32         //String(byte[] bytes,String charsetName):通過指定的字符集解碼指定的位元組陣列來構造新的String
33 //        String ss=new String(bys,"UTF-8");//?й?
34         String ss=new String(bys,"GBK");//中國
35         System.out.println(ss);
36     }
37 }

字元流中的編碼解碼

字元流抽象基類:

    • Reader:字元輸入流的抽象類
    • Writer:字元輸出流的抽象類

字元流中和編碼解碼的兩個類:

    • InputStreamReader:是從位元組流到字元流的橋樑

它讀取位元組,並使用指定的編碼將其解碼為字元
它使用的字符集可以由名稱指定,也可以被明確指定,或者可以接收平臺的預設字符集

    • OutputStreamWriter:是從字元流到位元組流的橋樑

是從字元流到位元組流的橋樑,使用指定的編碼將寫入的字元編碼為位元組
它使用的字符集可以由名稱指定,也可以被明確指定,或者可以接收平臺的預設字符集

 1 import java.io.FileInputStream;
 2 import java.io.FileOutputStream;
 3 import java.io.IOException;
 4 import java.io.InputStreamReader;
 5 import java.io.OutputStreamWriter;
 6 
 7 /*
 8 - InputStreamReader:是從位元組流到字元流的橋樑
 9     它讀取位元組,並使用指定的編碼將其解碼為字元
10     它使用的字符集可以由名稱指定,也可以被明確指定,或者可以接收平臺的預設字符集
11 
12 - OutputStreamWriter:是從字元流到位元組流的橋樑
13     是從字元流到位元組流的橋樑,使用指定的編碼將寫入的字元編碼為位元組
14     它使用的字符集可以由名稱指定,也可以被明確指定,或者可以接收平臺的預設字符集
15 */
16 public class ConversionStreamDemo {
17     public static void main(String[] args) throws IOException{
18         //OutputStreamWriter(OutputStream out)建立一個使用預設字元編碼的OutputStreamWriter
19         //OutputStreamWriter(OutputStream out,String charsetName)建立一個使用命名字符集的OutputStreamWriter
20         
21 //        FileOutputStream fos=new FileOutputStream("..\\hello java\\osw.txt");
22 //        OutputStreamWriter osw=new OutputStreamWriter(fos);
23         //兩步合併
24 //        OutputStreamWriter osw=new OutputStreamWriter(new FileOutputStream("..\\hello java\\osw.txt"));
25 //        OutputStreamWriter osw=new OutputStreamWriter(new FileOutputStream("..\\hello java\\osw.txt"),"GBK");//中國
26         OutputStreamWriter osw=new OutputStreamWriter(new FileOutputStream("..\\hello java\\osw.txt"),"UTF-8");//涓浗
27         osw.write("中國");
28         osw.close();
29         
30         //InputStreamReader(InputStream in)建立一個使用預設字符集的InputStreamReader
31         //InputStreamReader(InputStream in,String charsetName)建立一個使用命名字符集的InputStreamReader
32         
33 //        InputStreamReader isr=new InputStreamReader(new FileInputStream("..\\hello java\\osw.txt"));//涓浗
34 //        InputStreamReader isr=new InputStreamReader(new FileInputStream("..\\hello java\\osw.txt"),"GBK");//涓浗
35         InputStreamReader isr=new InputStreamReader(new FileInputStream("..\\hello java\\osw.txt"),"UTF-8");//中國
36         //一次讀取一個字元資料
37         int ch;
38         while((ch=isr.read())!=-1) {
39             System.out.print((char)ch);
40         }
41         
42         isr.close();
43         
44     }
45 }

字元流寫資料的5種方式

 1 import java.io.FileOutputStream;
 2 import java.io.IOException;
 3 import java.io.OutputStreamWriter;
 4 
 5 /*
 6 構造方法:
 7     OutputStreamWriter(OutputStream out)建立一個使用預設字元編碼的OutputStreamWriter
 8     
 9     字元流寫資料的5種方式**
10 
11     void write(int c)   寫一個字元
12     void write(char[] cbuf)寫入一個字元陣列
13     void write(char[] cbuf,int off,int len)寫入字元陣列的一部分
14     void write(String str)寫一個字串
15     void write(String str,int off,int len)寫一個字串的一部分
16 */
17 
18 public class OutputStreamWriterDemo {
19     public static void main(String[] args) throws IOException {
20         //        OutputStreamWriter(OutputStream out)建立一個使用預設字元編碼的OutputStreamWriter
21         OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("..\\hello java\\osw.txt"));
22         /*
23         //void write(int c)   寫一個字元
24         osw.write(97);//不重新整理資料進不來,因為真正底層寫資料是位元組流,而字元流相對於位元組流是有緩衝的
25         //void flush()重新整理流
26         osw.flush();
27         osw.write(98);
28         osw.flush();
29         osw.write(99);*/
30 
31         //void write(char[] cbuf)寫入一個字元陣列
32         char[] chs = { 'a', 'b', 'c', 'd', 'e' };
33         //osw.write(chs);//abcde
34 
35         //void write(char[] cbuf,int off,int len)寫入字元陣列的一部分
36         //osw.write(chs,0,chs.length);//abcde
37         //osw.write(chs,1,3);//bcd
38 
39         //void write(String str)寫一個字串
40         //osw.write("abcde");//abcde
41 
42         //void write(String str,int off,int len)寫一個字串的一部分
43         //osw.write("abcde",0,"abcde".length());//abcde
44         osw.write("abcde", 1, 3);//bcd
45 
46         osw.close();//關閉流,先重新整理
47         //osw.write(100);//java.io.IOException: Stream closed
48 
49     }
50 }

字元流讀資料的2種方式

 1 import java.io.FileInputStream;
 2 import java.io.IOException;
 3 import java.io.InputStreamReader;
 4 
 5 /*
 6 構造方法:
 7     InputStreamReader(InputStream in)建立一個使用預設字符集的InputStreamReader
 8     
 9 字元流讀資料的2種方式**
10     int read()  一次讀一個字元資料
11     int read(char[] cbuf)一次讀一個字元陣列資料
12 
13 */
14 public class InputStreamReaderDemo {
15     public static void main(String[] args) throws IOException{
16         //InputStreamReader(InputStream in)建立一個使用預設字符集的InputStreamReader
17 //        InputStreamReader isr=new InputStreamReader(new FileInputStream("..\\hello java\\osw.txt"));
18         InputStreamReader isr=new InputStreamReader(new FileInputStream("..\\hello java\\ConversionStreamDemo.java"));
19         
20         
21         //int read()  一次讀一個字元資料
22 //        int ch;
23 //        while((ch=isr.read())!=-1) {
24 //            System.out.print((char)ch);
25 //        }
26         /*執行結果:
27         hello
28         world
29         java*/
30         
31         //int read(char[] cbuf)一次讀一個字元陣列資料
32         char[] chs=new char[1024];
33         int len;
34         while((len=isr.read(chs))!=-1) {
35             System.out.print(new String(chs,0,len));
36         }
37         /*執行結果:
38         hello
39         world
40         java*/
41         
42         //釋放資源
43         isr.close();
44         
45     }
46 }

案例:複製Java檔案

需求:把模組目錄下的“ConversionSteamDemo.java”複製到模組目錄下的“Copy.java”

思路:

  1. 根據資料來源建立字元輸入流物件
  2. 根據目的地建立字元輸出流物件
  3. 讀寫資料,複製檔案
  4. 釋放資源
 1 import java.io.FileInputStream;
 2 import java.io.FileOutputStream;
 3 import java.io.IOException;
 4 import java.io.InputStreamReader;
 5 import java.io.OutputStreamWriter;
 6 
 7 /*
 8 案例:複製Java檔案
 9 需求:把模組目錄下的“ConversionSteamDemo.java”複製到模組目錄下的“Copy.java”
10 
11 思路:
12 
13  1. 根據資料來源建立字輸入流物件
14  2. 根據目的地建立字元輸出流物件
15  3. 讀寫資料,複製檔案
16  4. 釋放資源
17 */
18 
19 public class CopyJavaDemo {
20     public static void main(String[] args) throws IOException {
21         //1. 根據資料來源建立字輸入流物件
22         InputStreamReader isr = new InputStreamReader(new FileInputStream("..\\hello java\\ConversionStreamDemo.java"));
23         //2. 根據目的地建立字元輸出流物件
24         OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("..\\hello java\\Copy.java"));
25         //3. 讀寫資料,複製檔案
26         //一次讀寫一個字元資料
27         //int ch;
28         //while((ch=isr.read())!=-1) {
29         //    osw.write(ch);
30         //}
31 
32         //一次讀寫一個字元陣列資料
33         char[] chs = new char[1024];
34         int len;
35         while ((len = isr.read(chs)) != -1) {
36             osw.write(chs, 0, len);
37         }
38 
39         //釋放資源
40         isr.close();
41         osw.close();
42 
43     }
44 }

複製Java檔案(改進版)

轉換流的名字比較長,而我們常見的操作都是按照本地預設編碼實現的,所以,為了簡化書寫,轉換流提供了對應的子類

  • FileReader:用於讀取字元檔案的便捷類

FileReader(String fileName)

  • FileWriter:用於寫入字元檔案的便捷類

FileWriter(String fileName)

資料來源和目的地的分析

資料來源:…\hello java\ConversionStreamDemo.java—讀資料—Reader—InputStreamReader—FileReader

目的地:…\hello java\Copy.java–寫資料–Writer–OutputStreamWriter—FileWriter

 1 import java.io.FileReader;
 2 import java.io.FileWriter;
 3 import java.io.IOException;
 4 
 5 /*
 6 案例:複製Java檔案(改進版)
 7 
 8 需求:把模組目錄下的“ConversionSteamDemo.java”複製到模組目錄下的“Copy.java”
 9 
10 分析:
11 
12  1. 轉換流的名字比較長,而我們常見的操作都是按照本地預設編碼實現的,所以,為了簡化書寫,轉換流提供了對應的子類
13  2. FileReader:用於讀取字元檔案的便捷類
14     FileReader(String fileName)
15  3. FileWriter:用於寫入字元檔案的便捷類
16     FileWriter(String fileName)
17  4. 資料來源和目的地的分析
18     資料來源:---讀資料---Reader---InputStreamReader---FileReader
19     目的地:..\\hello java\\Copy.java--寫資料--Writer--OutputStreamWriter---FileWriter
20 
21 思路:
22 
23 1. 根據資料來源建立字輸入流物件
24 2. 根據目的地建立字元輸出流物件
25 3. 讀寫資料,複製檔案
26 4. 釋放資源
27 
28 */
29 public class CopyJavaDemo2 {
30     public static void main(String[] args) throws IOException {
31         //1. 根據資料來源建立字輸入流物件
32         FileReader fr = new FileReader("..\\hello java\\ConversionStreamDemo.java");
33 
34         //2. 根據目的地建立字元輸出流物件
35         FileWriter fw = new FileWriter("..\\hello java\\Copy.java");
36 
37         //3. 讀寫資料,複製檔案
38         //一次讀寫一個字元資料
39         //int ch;
40         //while((ch=fr.read())!=-1) {
41         //    fw.write(ch);
42         //}
43 
44         //一次讀寫一個字元陣列資料
45         char[] chs = new char[1024];
46         int len;
47         while ((len = fr.read(chs)) != -1) {
48             fw.write(chs, 0, len);
49         }
50 
51         //4. 釋放資源
52         fr.close();
53         fw.close();
54 
55     }
56 }

字元緩衝流

字元緩衝流:

    • BufferedWriter:將文字寫入字元輸出流,緩衝字元,以提供單個字元,陣列和字串的高效寫入,可以指定緩衝區大小,或者可以接收預設大小。預設值足夠大,可用於大多數用途
    • BufferedReader:從字元輸入流讀取文字,緩衝字元,以提供字元,陣列和行的高效讀取,可以指定緩衝區大小,或者可以使用預設大小。預設值足夠大,可用於大多數用途

構造方法:

    • BufferedWriter(Writer out)
    • BufferedReader(Reader in)

 

 1 import java.io.BufferedReader;
 2 import java.io.BufferedWriter;
 3 import java.io.FileReader;
 4 import java.io.FileWriter;
 5 import java.io.IOException;
 6 
 7 /*
 8 字元緩衝流:
 9 
10 - BufferedWriter:將文字寫入字元輸出流,緩衝字元,以提供單個字元,陣列和字串的高效寫入,可以指定緩衝區大小,或者可以接收預設大小。預設值足夠大,可用於大多數用途
11 - BufferedReader:從字元輸入流讀取文字,緩衝字元,以提供字元,陣列和行的高效讀取,可以指定緩衝區大小,或者可以使用預設大小。預設值足夠大,可用於大多數用途
12 
13 構造方法:
14 
15 - BufferedWriter(Writer out)
16 - BufferedReader(Reader in)
17 */
18 
19 public class BufferedStreamDemo2 {
20     public static void main(String[] args) throws IOException {
21         //BufferedWriter(Writer out)
22         //        FileWriter fw=new FileWriter("..\\hello java\\bw.txt");
23         //        BufferedWriter bw=new BufferedWriter(fw);
24         /*
25         //可合併為一句
26         BufferedWriter bw=new BufferedWriter(new FileWriter("..\\hello java\\bw.txt"));
27         
28         bw.write("hello\r\n");
29         bw.write("world\r\n");
30         
31         bw.close();*/
32 
33         //BufferedReader(Reader in)
34         BufferedReader br = new BufferedReader(new FileReader("..\\hello java\\bw.txt"));
35         /*
36         //一次讀一個字元資料
37         int ch;
38         while((ch=br.read())!=-1) {
39             System.out.print((char)ch);
40         }*/
41 
42         //一次讀一個字元陣列資料
43         char[] chs = new char[1024];
44         int len;
45         while ((len = br.read(chs)) != -1) {
46             System.out.print(new String(chs, 0, len));
47         }
48 
49         br.close();
50 
51     }
52 }

複製Java檔案(字元緩衝流改進版)

 1 import java.io.BufferedReader;
 2 import java.io.BufferedWriter;
 3 import java.io.FileReader;
 4 import java.io.FileWriter;
 5 import java.io.IOException;
 6 
 7 /*
 8 案例:複製Java檔案(字元緩衝流改進版)
 9 
10 需求:把模組目錄下的“ConversionStreamDemo.java”複製到模組目錄下的“Copy.java”
11 
12 思路:
13     1. 根據資料來源建立字元緩衝輸入流物件
14     2. 根據目的地建立字元緩衝輸出流物件
15     3. 讀寫資料,複製檔案
16     4. 釋放資源
17 */
18 
19 public class CopyJavaDemo3 {
20     public static void main(String[] args) throws IOException {
21         //1. 根據資料來源建立字元緩衝輸入流物件
22         BufferedReader br = new BufferedReader(new FileReader("..\\hello java\\ConversionStreamDemo.java"));
23 
24         //2. 根據目的地建立字元緩衝輸出流物件
25         BufferedWriter bw = new BufferedWriter(new FileWriter("..\\hello java\\Copy.java"));
26 
27         //3. 讀寫資料,複製檔案
28         /*
29         //一次讀寫一個字元資料
30         int ch;
31         while((ch=br.read())!=-1) {
32             bw.write(ch);
33         }*/
34 
35         //一次讀寫一個字元陣列資料
36         char[] chs = new char[1024];
37         int len;
38         while ((len = br.read(chs)) != -1) {
39             bw.write(chs, 0, len);
40         }
41 
42         //4. 釋放資源
43         br.close();
44         bw.close();
45 
46     }
47 }

字元緩衝流的特有功能

  • BufferedWrite:

void newLine():寫行分隔符,行分隔符字串有系統屬性定義

  • BufferedReader:

public String readLine():讀一行文字。結果包含行的內容的字串,不包括任何行終止字元,如果流的結尾已經到大,則為null

 1 /*
 2 字元緩衝流的特有功能
 3 
 4 BufferedWrite:
 5     void newLine():寫一行行分隔符,行分隔符字串有系統屬性定義
 6 
 7 BufferedReader:
 8     public String readLine():讀一行文字。結果包含行的內容的字串,不包括任何行終止字元,如果流的結尾已經到大,則為null
 9 
10 */
11 
12 import java.io.BufferedReader;
13 import java.io.BufferedWriter;
14 import java.io.FileReader;
15 import java.io.FileWriter;
16 import java.io.IOException;
17 
18 public class BufferedStreamDemo3 {
19     public static void main(String[] args) throws IOException {
20         /*//建立字元緩衝輸出流
21         BufferedWriter bw=new BufferedWriter(new FileWriter("..\\hello java\\bw.txt"));
22         
23         //寫資料
24         for(int i=0;i<10;i++) {
25             bw.write("hello"+i);
26         //            bw.write("\r\n");//只適用於Windows
27             bw.newLine();
28             bw.flush();
29         } 
30             hello0
31             hello1
32             hello2
33             hello3
34             hello4
35             hello5
36             hello6
37             hello7
38             hello8
39             hello9
40         
41         //釋放資源
42         bw.close();
43         */
44 
45         //建立字元緩衝輸入流
46         BufferedReader br = new BufferedReader(new FileReader("..\\hello java\\bw.txt"));
47 
48         //public String readLine():讀一行文字。結果包含行的內容的字串,不包括任何行終止字元,如果流的結尾已經到大,則為null
49         /*//第一次讀取資料
50         String line = br.readLine();
51         System.out.println(line);//hello0
52         //第二次讀取資料
53         line = br.readLine();
54         System.out.println(line);//hello1
55         //再多讀兩次
56         line = br.readLine();
57         System.out.println(line);//null
58         line = br.readLine();
59         System.out.println(line);//null
60         */
61 
62         String line;
63         while ((line = br.readLine()) != null) {
64             System.out.println(line);//readline只讀內容,不包含換行符,要加ln
65         }
66 
67         //釋放資源
68         br.close();
69 
70     }
71 }

複製Java檔案(字元緩衝流特有功能改進版)

 

 1 import java.io.BufferedReader;
 2 import java.io.BufferedWriter;
 3 import java.io.FileReader;
 4 import java.io.FileWriter;
 5 import java.io.IOException;
 6 
 7 /*
 8 案例:複製Java檔案(字元緩衝流特有功能改進版)
 9 
10 需求:把模組目錄下的“ConversionSteamDemo.java”複製到模組目錄下的“Copy.java”
11 
12 思路:
13 
14  1. 根據資料來源建立字元緩衝輸入流物件
15  2. 根據目的地建立字元緩衝輸出流物件
16  3. 讀寫資料,複製檔案
17          使用字元緩衝流特有功能實現
18  4. 釋放資源
19 */
20 public class CopyJavaDemo4 {
21     public static void main(String[] args) throws IOException {
22         //1. 根據資料來源建立字元緩衝輸入流物件
23         BufferedReader br = new BufferedReader(new FileReader("..\\hello java\\ConversionStreamDemo.java"));
24 
25         //2. 根據目的地建立字元緩衝輸出流物件
26         BufferedWriter bw = new BufferedWriter(new FileWriter("..\\hello java\\Copy.java"));
27 
28         //3. 讀寫資料,複製檔案(使用字元緩衝流特有功能實現)
29         String line;
30         while ((line = br.readLine()) != null) {
31             bw.write(line);
32             bw.newLine();//要加換行
33             bw.flush();
34         }
35 
36         //4. 釋放資源
37         br.close();
38         bw.close();
39 
40     }
41 }

IO流小結

 

 

複製檔案的異常處理

try…catch…finally的做法:

1 try{
2     可能出現異常的程式碼;
3     }catch(異常類名 變數名){
4         異常的處理程式碼;
5     }finally{
6         執行所有清除操作;
7     }

JDK7改進方案:

1 try(定義流物件){
2     可能出現異常的程式碼;
3 }catch(異常類名 變數名){
4     異常的處理程式碼;
5 }
6 
7 自動釋放資源

JDK9的改進方案:

1 定義輸入流物件;
2 定義輸出流物件;
3 try(輸入流物件;輸出流物件){
4     可能出現異常的程式碼;
5 }catch(異常類名 變數名){
6     異常的處理程式碼;
7 }
8 
9 自動釋放資源
 1 import java.io.FileReader;
 2 import java.io.FileWriter;
 3 import java.io.IOException;
 4 
 5 /*
 6 複製檔案加入異常處理
 7 */
 8 public class CopyFileDemo {
 9     public static void main(String[] args) {
10 
11     }
12 
13     //JDK9的改進方案
14     private static void method4() throws IOException {
15         FileReader fr = new FileReader("fr.txt");
16         FileWriter fw = new FileWriter("fw.txt");
17         try (fr; fw) {
18 
19             char[] chs = new char[1024];
20             int len;
21             while ((len = fr.read()) != -1) {
22                 fw.write(chs, 0, len);
23             }
24         } catch (IOException e) {
25             e.printStackTrace();
26         }
27 
28     }
29 
30     //JDK7的改進方案
31     private static void method3() {
32         try (FileReader fr = new FileReader("fr.txt"); FileWriter fw = new FileWriter("fw.txt");) {
33 
34             char[] chs = new char[1024];
35             int len;
36             while ((len = fr.read()) != -1) {
37                 fw.write(chs, 0, len);
38             }
39         } catch (IOException e) {
40             e.printStackTrace();
41         }
42 
43     }
44 
45     //try...catch...finally
46     private static void method2() {
47         FileReader fr = null;//放外面定義,並給初始化
48         FileWriter fw = null;
49         try {
50             fr = new FileReader("fr.txt");
51             fw = new FileWriter("fw.txt");
52 
53             char[] chs = new char[1024];
54             int len;
55             while ((len = fr.read()) != -1) {
56                 fw.write(chs, 0, len);
57             }
58 
59         } catch (IOException e) {
60             e.printStackTrace();
61         } finally {//close出現問題,先判斷不為空,再用try...catch處理
62             if (fw != null) {
63 
64                 try {
65                     fw.close();
66                 } catch (IOException e) {
67                     // TODO Auto-generated catch block
68                     e.printStackTrace();
69                 }
70             }
71             //同理
72             if (fr != null) {
73                 try {
74                     fr.close();
75                 } catch (IOException e) {
76                     // TODO Auto-generated catch block
77                     e.printStackTrace();
78                 }
79             }
80 
81         }
82     }
83 
84     //丟擲異常
85     private static void method1() throws IOException {
86         FileReader fr = new FileReader("fr.txt");
87         FileWriter fw = new FileWriter("fw.txt");
88 
89         char[] chs = new char[1024];
90         int len;
91         while ((len = fr.read()) != -1) {
92             fw.write(chs, 0, len);
93         }
94 
95         fr.close();
96         fw.close();
97     }
98 
99 }