• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            奇奇的空間

            奇奇的空間
            posts - 5, comments - 13, trackbacks - 0, articles - 23

            Java的輸入輸出流

            Posted on 2006-05-08 16:52 奇奇 閱讀(203) 評論(0)  編輯 收藏 引用 所屬分類: JAVA

            Java語言的輸入輸出功能是十分強大而靈活的,美中不足的是看上去輸入輸出的代碼并不是很簡潔,因為你往往需要包裝許多不同的對象。在Java類庫中,IO部分的內容是很龐大的,因為它涉及的領域很廣泛:標準輸入輸出,文件的操作,網絡上的數據流,字符串流,對象流,zip文件流....本文的目的是為大家做一個簡要的介紹。

              流是一個很形象的概念,當程序需要讀取數據的時候,就會開啟一個通向數據源的流,這個數據源可以是文件,內存,或是網絡連接。類似的,當程序需要寫入數據的時候,就會開啟一個通向目的地的流。這時候你就可以想象數據好像在這其中“流”動一樣,如下圖:


              Java中的流分為兩種,一種是字節(jié)流,另一種是字符流,分別由四個抽象類來表示(每種流包括輸入和輸出兩種所以一共四個):InputStream,OutputStream,Reader,Writer。Java中其他多種多樣變化的流均是由它們派生出來的:



              在這其中InputStream和OutputStream在早期的Java版本中就已經存在了,它們是基于字節(jié)流的,而基于字符流的Reader和Writer是后來加入作為補充的。以上的層次圖是Java類庫中的一個基本的層次體系。

              在這四個抽象類中,InputStream和Reader定義了完全相同的接口:

            int read()
            int read(char cbuf[])
            int read(char cbuf[], int offset, int length)

              而OutputStream和Writer也是如此:

            int write(int c)
            int write(char cbuf[])
            int write(char cbuf[], int offset, int length)

              這六個方法都是最基本的,read()和write()通過方法的重載來讀寫一個字節(jié),或者一個字節(jié)數組。

              更多靈活多變的功能是由它們的子類來擴充完成的。知道了Java輸入輸出的基本層次結構以后,本文在這里想給大家一些以后可以反復應用例子,對于所有子類的細節(jié)及其功能并不詳細討論。

            import java.io.*;

              public class IOStreamDemo {

              ??????public void samples() throws IOException {

              ???????????//1. 這是從鍵盤讀入一行數據,返回的是一個字符串
              ???????????BufferedReader stdin =new BufferedReader(new InputStreamReader(System.in));
              ???????????System.out.print("Enter a line:");
              ???????????System.out.println(stdin.readLine());

              ???????????//2. 這是從文件中逐行讀入數據

              ???????????BufferedReader in = new BufferedReader(new FileReader("IOStreamDemo.java"));
              ???????????String s, s2 = new String();
              ???????????while((s = in.readLine())!= null
              ??????????????????????s2 += s + "\n"
              ???????????in.close();

              ???????????//3. 這是從一個字符串中逐個讀入字節(jié)
              ???????????StringReader in1 = new StringReader(s2);
              ???????????int c;
              ???????????while((c = in1.read()) != -1)
              ??????????????????????System.out.print((char)c);

              ???????????//4. 這是將一個字符串寫入文件
              ???????????try {
              ??????????????????????BufferedReader in2 = new BufferedReader(new StringReader(s2));
              ??????????????????????PrintWriter out1 = new PrintWriter(new BufferedWriter(new FileWriter("IODemo.out")));
              ??????????????????????int lineCount = 1;
              ??????????????????????while((s = in2.readLine()) != null
              ?????????????????????????????????out1.println(lineCount++ + ": " + s);
              ??????????????????????out1.close();
              ???????????} catch(EOFException e) {
              ??????????????????????System.err.println("End of stream");
              ???????????}
              ??????}

              }

              對于上面的例子,需要說明的有以下幾點:

              1. BufferedReaderReader的一個子類,它具有緩沖的作用,避免了頻繁的從物理設備中讀取信息。它有以下兩個構造函數:

            BufferedReader Reader in)
            BufferedReaderReader in, int sz)

              這里的sz是指定緩沖區(qū)的大小。

              它的基本方法:

            void close(//關閉流

            ???????????void mark(int readAheadLimit) //標記當前位置

            ???????????boolean markSupported() //是否支持標記

            ???????????int read() //繼承自Reader的基本方法

            ???????????int read(char[] cbuf, int off,int len) //繼承自Reader的基本方法

            ???????????String readLine() //讀取一行內容并以字符串形式返回

            ???????????boolean ready() //判斷流是否已經做好讀入的準備

            ???????????void reset() //重設到最近的一個標記

            ???????????long skip(long n) //跳過指定個數的字符讀取

              2. InputStreamReaderInputStreamReader之間的橋梁,由于System.in是字節(jié)流,需要用它來包裝之后變?yōu)樽址鞴┙o ????????????BufferedReader使用。

              3. PrintWriter out1 = new PrintWriter(new BufferedWriter(new FileWriter("IODemo.out")));

              這句話體現了Java輸入輸出系統的一個特點,為了達到某個目的,需要包裝好幾層。首先,輸出目的地是文件IODemo.out,所以最內層包裝的是FileWriter,建立一個輸出文件流,接下來,我們希望這個流是緩沖的,所以用BufferedWriter來包裝它以達到目的,最后,我們需要格式化輸出結果,于是將PrintWriter包在最外層。

              Java提供了這樣一個功能,將標準的輸入輸出流轉向,也就是說,我們可以將某個其他的流設為標準輸入或輸出流,看下面這個例子:

            import java.io.*;

            public class Redirecting {

            ???????public static void
            main(String[] args) throws IOException {
            ??????????????PrintStream console = System.out;
            ??????????????BufferedInputStream in = newBufferedInputStreamnew FileInputStream"Redirecting.java"));
            ??????????????PrintStream out = newPrintStream( new BufferedOutputStream new FileOutputStream"test.out")));
            ??????????????System.setIn(in);
            ??????????????System.setOut(out);

            ??????????????BufferedReader br = newBufferedReadernewInputStreamReader(System.in));
            ??????????????String s;
            ??????????????while((s = br.readLine()) != null
            ?????????????????????System.out.println(s);
            ??????????????out.close();
            ??????????????System.setOut(console);
            ??????}
            }

              在這里java.lang.System的靜態(tài)方法

            static void setIn(InputStream in)
            static void setOut(PrintStream out)

              提供了重新定義標準輸入輸出流的方法,這樣做是很方便的,比如一個程序的結果有很多,有時候甚至要翻頁顯示,這樣不便于觀看結果,這是你就可以將標準輸出流定義為一個文件流,程序運行完之后打開相應的文件觀看結果,就直觀了許多。

              Java流有著另一個重要的用途,那就是利用對象流對對象進行序列化。下面將開始介紹這方面的問題。

              在一個程序運行的時候,其中的變量數據是保存在內存中的,一旦程序結束這些數據將不會被保存,一種解決的辦法是將數據寫入文件,而Java中提供了一種機制,它可以將程序中的對象寫入文件,之后再從文件中把對象讀出來重新建立。這就是所謂的對象序列化Java中引入它主要是為了RMI(Remote Method Invocation)和Java Bean所用,不過在平時應用中,它也是很有用的一種技術。

              所有需要實現對象序列化的對象必須首先實現Serializable接口。下面看一個例子:

            import java.io.*;
            import java.util.*;

            public class Logon implements Serializable {

            ???????private Date date = new Date();
            ???????private String username;
            ???????private transient String password;


            ???????Logon(Stringname, String pwd) {
            ??????????????username = name;
            ??????????????password = pwd;
            ???????}


            ???????public String toString() {
            ??????????????String pwd = (password == null) ? "(n/a)" : password;
            ??????????????return "logon info: \n " + "username: " + username + "\n date: " + date + "\n password: " + pwd;
            ???????}


            ???????public static void
            main(String[] args) throws IOException, ClassNotFoundException {
            ??????????????Logon a = new Logon("Morgan", "morgan83");
            ??????????????System.out.println( "logon a = " + a);
            ??????????????ObjectOutputStream o = newObjectOutputStreamnewFileOutputStream"Logon.out"));
            ??????????????o.writeObject(a);
            ??????????????o.close();

            ??????????????int seconds = 5;
            ??????????????long t = System.currentTimeMillis()?+ seconds * 1000;
            ??????????????whileSystem.currentTimeMillis() < t) ;

            ??????????????ObjectInputStream in = new ObjectInputStreamnewFileInputStream"Logon.out"));
            ??????????????System.out.println( "Recovering object at " + new Date());
            ??????????????a = (Logon)in.readObject();
            ??????????????System.out.println("logon a = " + a);
            ???????}
            }

              類Logon是一個記錄登錄信息的類,包括用戶名和密碼。首先它實現了接口Serializable,這就標志著它可以被序列化。之后再main方法里ObjectOutputStream o = newObjectOutputStreamnewFileOutputStream"Logon.out"));新建一個對象輸出流包裝一個文件流,表示對象序列化的目的地是文件Logon.out。然后用方法writeObject開始寫入。想要還原的時候也很簡單ObjectInputStream in = newObjectInputStreamnewFileInputStream"Logon.out"));新建一個對象輸入流以文件流Logon.out為參數,之后調用readObject方法就可以了。

              需要說明一點,對象序列化有一個神奇之處就是,它建立了一張對象網,將當前要序列化的對象中所持有的引用指向的對象都包含起來一起寫入到文件,更為奇妙的是,如果你一次序列化了好幾個對象,它們中相同的內容將會被共享寫入。這的確是一個非常好的機制。它可以用來實現深層拷貝。

              關鍵字transient在這里表示當前內容將不被序列化,比如例子中的密碼,需要保密,所以沒有被寫入文

            国内精品久久久久久久亚洲| 99久久精品国产一区二区| 国产精品99久久久久久人| 久久久久久a亚洲欧洲aⅴ| 久久九色综合九色99伊人| 久久久国产99久久国产一| 久久精品国产亚洲精品2020| 免费国产99久久久香蕉| 精品久久久无码21p发布| 四虎国产精品免费久久久| 亚洲精品午夜国产va久久| 国产精品久久国产精品99盘| 久久久99精品成人片中文字幕| 久久精品免费一区二区| 久久久久18| 久久婷婷五月综合国产尤物app| 伊人色综合久久天天| 午夜欧美精品久久久久久久| 国产精品成人99久久久久| 久久亚洲春色中文字幕久久久| 久久免费视频6| 国产成人无码精品久久久免费| 欧美精品久久久久久久自慰| 久久精品视频一| 欧美性猛交xxxx免费看久久久| 久久亚洲综合色一区二区三区| 亚洲国产另类久久久精品| 久久亚洲AV无码西西人体| 99久久伊人精品综合观看| 99久久国语露脸精品国产| 性欧美丰满熟妇XXXX性久久久 | 久久这里只精品国产99热| 奇米综合四色77777久久| 香蕉久久久久久狠狠色| 伊人精品久久久久7777| 久久久无码精品午夜| 久久精品中文字幕有码| 久久中文字幕视频、最近更新| 国产69精品久久久久9999| 99久久精品费精品国产| 91性高湖久久久久|