[十三]JavaIO之PushBackInputStream

功能簡介
PushBackInputStream是針對於輸入的一種擴展功能
裝飾器模式中的具體的裝飾類,抽象的裝飾器爲FilterInputStream
PushBackInputStream的重點在於理解緩衝區的使用手段

image_5b98a4e3_571f


流自己不支持回退功能,想要可以pushBack 顯然必須可以緩存數據
PushBackInputStream內部維護了一個字節數組
image_5b98a4e3_5e86
不要想固然的認爲,相似BufferedInputStream,內部緩衝了一個數組,全部的數據都要通過這個緩衝區,而後對他作處理
他內部維護的緩衝區,僅僅保存pushBack的字節
還須要注意的是他的內部緩衝區是從後往前寫入的,也就是下圖中的,從右往左
下標索引大的先寫入

一旦寫滿, 會拋出異常
throw new IOException("Push back buffer is full");
 
image_5b98a4e3_4039


image_5b98a4e3_3bda
推回一個字節
因爲是從最後開始,一旦pos等於0也就是到了最前面,就無處可放了
因此拋出異常
不然,就在前面寫入一個
image_5b98a4e3_d7b
推回 指定字節數組b 從off偏移量開始的len個字節
推回len個長度,既然是從後往前設置數據,那麼pos的值就是可用 空間 的個數
若是空間不夠,拋出異常
不然,pos直接向前移動到指定位置
而後藉助於System.arraycopy 將b 便宜量off開始拷貝到
image_5b98a4e3_1701
是unread(byte[] b, int off, int len)的簡化形式
用於拷貝整個數組

從unread能夠清晰的看出來
內部的緩衝區,只是爲了給被回退掉的數據使用的
每次回退就是在最前面(下表索引較小的位置)  寫入數據



read
只要明確了unread以及內部緩衝區的存放形式便可很清楚的知道read的工做方式了
image_5b98a4e3_5b63
既然是從最後開始往前依次存放,初始時 pos == buf.length  那麼但凡是有數據,pos必然,小於buf.length
也就是說,讀取一個字節時,若是回退緩衝內有數據,直接返回一個字節
不然,調用父類的read方法,而父類FilterInputStream 只是簡單的 封裝轉發請求
實際上就是直接使用內部的InputStream進行處理

多參數的read方法,將數據寫入到指定的字節數組b中

回退緩衝區中有效字節個數爲avail = buf.length - pos;   
若是緩衝區中有數據,那麼將首先從回退緩衝區中讀取數據

若是讀取的字節數少於 len 個字節,那麼它將從底層輸入流中讀取字節


image_5b98a4e3_16d1


PushBackInputStream 不支持標記點相關的操做
image_5b98a4e3_87b



總結

PushBackInputStream的實現邏輯比較清晰
我內部有一個緩衝區,你要是回退了我就往裏面寫入數據
每次的數據讀取都是先看看緩衝區裏面有沒有數據,有的話就先讀取回退緩衝區的
不然,就再去使用實際的流去進行讀取
看得出來,若是你歷來未曾回退過,那麼好像什麼都同樣,仍是使用原來的InputStream 進行讀取
相關文章
相關標籤/搜索