Java入門 - 語言基礎 - 20.Stream和File和IO

原文地址:http://www.work100.net/training/java-stream-file-io.html
更多教程:光束雲 - 免費課程

Stream和File和IO

序號 文內章節 視頻
1 概述 -
2 流(Stream) -
3 讀寫文件 -
4 文件和I/O -

請參照如上章節導航進行閱讀

1.概述

Java.io 包幾乎包含了所有操作輸入、輸出需要的類。所有這些流類代表了輸入源和輸出目標。

Java.io 包中的流支持很多種格式,比如:基本類型、對象、本地化字符集等等。

一個流可以理解爲一個數據的序列。輸入流表示從一個源讀取數據,輸出流表示向一個目標寫數據。

Java 爲 I/O 提供了強大的而靈活的支持,使其更廣泛地應用到文件傳輸和網絡編程中。

但本節講述最基本的和流與 I/O 相關的功能。我們將通過一個個例子來學習這些功能。

2.流(Stream)

2.1.讀取控制檯輸入

Java 的控制檯輸入由 System.in 完成。

爲了獲得一個綁定到控制檯的字符流,你可以把 System.in 包裝在一個 BufferedReader 對象中來創建一個字符流。

下面是創建 BufferedReader 的基本語法:

BufferedReader br = new BufferedReader(new 
                      InputStreamReader(System.in));

BufferedReader 對象創建後,我們便可以使用 read() 方法從控制檯讀取一個字符,或者用 readLine() 方法讀取一個字符串。

2.2.從控制檯讀取多字符輸入

BufferedReader 對象讀取一個字符要使用 read() 方法,它的語法如下:

int read( ) throws IOException

每次調用 read() 方法,它從輸入流讀取一個字符並把該字符作爲整數值返回。 當流結束的時候返回 -1。該方法拋出 IOException

下面的程序示範了用 read() 方法從控制檯不斷讀取字符直到用戶輸入 "q"。

//使用 BufferedReader 在控制檯讀取字符

import java.io.*;

public class BRRead {
    public static void main(String[] args) throws IOException {
        char c;
        // 使用 System.in 創建 BufferedReader
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        System.out.println("輸入字符, 按下 'q' 鍵退出。");
        // 讀取字符
        do {
            c = (char) br.read();
            System.out.println(c);
        } while (c != 'q');
    }
}

以上實例編譯運行結果如下:

輸入字符, 按下 'q' 鍵退出。
work100
w
o
r
k
1
0
0

q
q

2.3.從控制檯讀取字符串

從標準輸入讀取一個字符串需要使用 BufferedReaderreadLine() 方法。

它的一般格式是:

String readLine( ) throws IOException

下面的程序讀取和顯示字符行直到你輸入了單詞"end"。

//使用 BufferedReader 在控制檯讀取字符
import java.io.*;

public class BRReadLines {
    public static void main(String[] args) throws IOException {
        // 使用 System.in 創建 BufferedReader
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        String str;
        System.out.println("Enter lines of text.");
        System.out.println("Enter 'end' to quit.");
        do {
            str = br.readLine();
            System.out.println(str);
        } while (!str.equals("end"));
    }
}

以上實例編譯運行結果如下:

Enter lines of text.
Enter 'end' to quit.
This is line one
This is line one
This is line two
This is line two
end
end

JDK 5 後的版本我們也可以使用 Java Scanner 類來獲取控制檯的輸入。

2.4.控制檯輸出

在此前已經介紹過,控制檯的輸出由 print()println() 完成。這些方法都由類 PrintStream 定義,System.out 是該類對象的一個引用。

PrintStream 繼承了 OutputStream 類,並且實現了方法 write()。這樣,write() 也可以用來往控制檯寫操作。

PrintStream 定義 write() 的最簡單格式如下所示:

void write(int byteval)

該方法將 byteval 的低八位字節寫到流中。

實例

下面的例子用 write() 把字符 "A" 和緊跟着的換行符輸出到屏幕:

import java.io.*;

//演示 System.out.write().
public class WriteDemo {
    public static void main(String[] args) {
        int b;
        b = 'A';
        System.out.write(b);
        System.out.write('\n');
    }
}

運行以上實例在輸出窗口輸出 "A" 字符

A

注意:write() 方法不經常使用,因爲 print()println() 方法用起來更爲方便。

3.讀寫文件

如前所述,一個流被定義爲一個數據序列。輸入流用於從源讀取數據,輸出流用於向目標寫數據。

下圖是一個描述輸入流和輸出流的類層次圖。

<img src="http://contents.work100.net/images/training/java/stream-file-io/iostream1.jpg" style="max-width: 98%" />

下面將要討論的兩個重要的流是 FileInputStreamFileOutputStream

3.1.FileInputStream

該流用於從文件讀取數據,它的對象可以用關鍵字 new 來創建。

有多種構造方法可用來創建對象。

可以使用字符串類型的文件名來創建一個輸入流對象來讀取文件:

InputStream f = new FileInputStream("C:/java/hello");

也可以使用一個文件對象來創建一個輸入流對象來讀取文件。我們首先得使用 File() 方法來創建一個文件對象:

File f = new File("C:/java/hello");
InputStream out = new FileInputStream(f);

創建了 InputStream 對象,就可以使用下面的方法來讀取流或者進行其他的流操作。

序號 方法及描述
1 public void close() throws IOException{} <br>關閉此文件輸入流並釋放與此流有關的所有系統資源。拋出 IOException 異常。
2 protected void finalize()throws IOException{} <br>這個方法清除與該文件的連接。確保在不再引用文件輸入流時調用其 close 方法。拋出 IOException 異常。
3 public int read(int r)throws IOException{} <br>這個方法從 InputStream 對象讀取指定字節的數據。返回爲整數值。返回下一字節數據,如果已經到結尾則返回 -1
4 public int read(byte[] r) throws IOException{} <br>這個方法從輸入流讀取 r.length 長度的字節。返回讀取的字節數。如果是文件結尾則返回 -1
5 public int available() throws IOException{} <br>返回下一次對此輸入流調用的方法可以不受阻塞地從此輸入流讀取的字節數。返回一個整數值。

3.2.FileOutputStream

該類用來創建一個文件並向文件中寫數據。

如果該流在打開文件進行輸出前,目標文件不存在,那麼該流會創建該文件。

有兩個構造方法可以用來創建 FileOutputStream 對象。

使用字符串類型的文件名來創建一個輸出流對象:

OutputStream f = new FileOutputStream("C:/java/hello")

也可以使用一個文件對象來創建一個輸出流來寫文件。我們首先得使用 File() 方法來創建一個文件對象:

File f = new File("C:/java/hello");
OutputStream f = new FileOutputStream(f);

創建 OutputStream 對象完成後,就可以使用下面的方法來寫入流或者進行其他的流操作。

序號 方法及描述
1 public void close() throws IOException{} <br>關閉此文件輸入流並釋放與此流有關的所有系統資源。拋出 IOException 異常。
2 protected void finalize()throws IOException{} <br>這個方法清除與該文件的連接。確保在不再引用文件輸入流時調用其 close 方法。拋出 IOException 異常。
3 public void write(int w)throws IOException{} <br>這個方法把指定的字節寫到輸出流中。
4 public void write(byte[] w) <br>把指定數組中 w.length 長度的字節寫到 OutputStream 中。

除了 OutputStream 外,還有一些其他的輸出流:

  • ByteArrayOutputStream
  • DataOutputStream

實例

下面是一個演示 InputStreamOutputStream 用法的例子:

import java.io.*;

public class fileStreamTest {
    public static void main(String[] args) {
        try {
            byte bWrite[] = { 11, 21, 3, 40, 5 };
            OutputStream os = new FileOutputStream("test.txt");
            for (int x = 0; x < bWrite.length; x++) {
                os.write(bWrite[x]); // writes the bytes
            }
            os.close();

            InputStream is = new FileInputStream("test.txt");
            int size = is.available();

            for (int i = 0; i < size; i++) {
                System.out.print((char) is.read() + "  ");
            }
            is.close();
        } catch (IOException e) {
            System.out.print("Exception");
        }
    }
}

上面的程序首先創建文件 test.txt,並把給定的數字以二進制形式寫進該文件,同時輸出到控制檯上。

以上代碼由於是二進制寫入,可能存在亂碼,你可以使用以下代碼實例來解決亂碼問題:

//文件名 :fileStreamTest2.java
import java.io.*;

public class fileStreamTest2 {
    public static void main(String[] args) throws IOException {

        File f = new File("a.txt");
        FileOutputStream fop = new FileOutputStream(f);
        // 構建FileOutputStream對象,文件不存在會自動新建

        OutputStreamWriter writer = new OutputStreamWriter(fop, "UTF-8");
        // 構建OutputStreamWriter對象,參數可以指定編碼,默認爲操作系統默認編碼,windows上是gbk

        writer.append("中文輸入");
        // 寫入到緩衝區

        writer.append("\r\n");
        // 換行

        writer.append("English");
        // 刷新緩存衝,寫入到文件,如果下面已經沒有寫入的內容了,直接close也會寫入

        writer.close();
        // 關閉寫入流,同時會把緩衝區內容寫入文件,所以上面的註釋掉

        fop.close();
        // 關閉輸出流,釋放系統資源

        FileInputStream fip = new FileInputStream(f);
        // 構建FileInputStream對象

        InputStreamReader reader = new InputStreamReader(fip, "UTF-8");
        // 構建InputStreamReader對象,編碼與寫入相同

        StringBuffer sb = new StringBuffer();
        while (reader.ready()) {
            sb.append((char) reader.read());
            // 轉成char加到StringBuffer對象中
        }
        System.out.println(sb.toString());
        reader.close();
        // 關閉讀取流

        fip.close();
        // 關閉輸入流,釋放系統資源
    }
}

4.文件和I/O

還有一些關於文件和 I/O 的類,我們也需要知道:

  • File Class(類)
  • FileReader Class(類)
  • FileWriter Class(類)

Java中的目錄

創建目錄:

File 類中有兩個方法可以用來創建文件夾:

  • mkdir() 方法創建一個文件夾,成功則返回 true,失敗則返回 false。失敗表明 File 對象指定的路徑已經存在,或者由於整個路徑還不存在,該文件夾不能被創建

  • mkdirs() 方法創建一個文件夾和它的所有父文件夾

下面的例子創建 "/tmp/user/java/bin"文件夾:

import java.io.File;

public class CreateDir {
    public static void main(String[] args) {
        String dirname = "/tmp/user/java/bin";
        File d = new File(dirname);
        // 現在創建目錄
        d.mkdirs();
    }
}

編譯並執行上面代碼來創建目錄 "/tmp/user/java/bin"。

注意: Java 在 UNIXWindows 自動按約定分辨文件路徑分隔符。如果你在 Windows 版本的 Java 中使用分隔符 (/) ,路徑依然能夠被正確解析。

讀取目錄

一個目錄其實就是一個 File 對象,它包含其他文件和文件夾。

如果創建一個 File 對象並且它是一個目錄,那麼調用 isDirectory() 方法會返回 true

可以通過調用該對象上的 list() 方法,來提取它包含的文件和文件夾的列表。

下面展示的例子說明如何使用 list() 方法來檢查一個文件夾中包含的內容:

import java.io.File;

public class DirList {
    public static void main(String[] args) {
        String dirname = "/tmp";
        File f1 = new File(dirname);
        if (f1.isDirectory()) {
            System.out.println("目錄 " + dirname);
            String s[] = f1.list();
            for (int i = 0; i < s.length; i++) {
                File f = new File(dirname + "/" + s[i]);
                if (f.isDirectory()) {
                    System.out.println(s[i] + " 是一個目錄");
                } else {
                    System.out.println(s[i] + " 是一個文件");
                }
            }
        } else {
            System.out.println(dirname + " 不是一個目錄");
        }
    }
}

以上實例編譯運行結果如下:

目錄 /tmp
bin 是一個目錄
lib 是一個目錄
demo 是一個目錄
test.txt 是一個文件
README 是一個文件
index.html 是一個文件
include 是一個目錄

刪除目錄或文件

刪除文件可以使用 java.io.File.delete() 方法。

以下代碼會刪除目錄 /tmp/java/,需要注意的是當刪除某一目錄時,必須保證該目錄下沒有其他文件才能正確刪除,否則將刪除失敗。

測試目錄結構:

/tmp/java/
|-- 1.log
|-- test
import java.io.File;

public class DeleteFileDemo {
    public static void main(String[] args) {
        // 這裏修改爲自己的測試目錄
        File folder = new File("/tmp/java/");
        deleteFolder(folder);
    }

    // 刪除文件及目錄
    public static void deleteFolder(File folder) {
        File[] files = folder.listFiles();
        if (files != null) {
            for (File f : files) {
                if (f.isDirectory()) {
                    deleteFolder(f);
                } else {
                    f.delete();
                }
            }
        }
        folder.delete();
    }
}

上一篇:方法
下一篇:Scanner類

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章