功能簡介
PushBackInputStream是針對於輸入的一種擴展功能
裝飾器模式中的具體的裝飾類,抽象的裝飾器爲FilterInputStream
PushBackInputStream的重點在於理解緩衝區的使用手段
流自己不支持回退功能,想要可以pushBack 顯然必須可以緩存數據 PushBackInputStream內部維護了一個字節數組 |
|
不要想固然的認爲,相似BufferedInputStream,內部緩衝了一個數組,全部的數據都要通過這個緩衝區,而後對他作處理
他內部維護的緩衝區,僅僅保存pushBack的字節
|
還須要注意的是他的內部緩衝區是從後往前寫入的,也就是下圖中的,從右往左 下標索引大的先寫入
一旦寫滿, 會拋出異常 throw new IOException("Push back buffer is full");
|
|
![image_5b98a4e3_3bda image_5b98a4e3_3bda](http://static.javashuo.com/static/loading.gif) 推回一個字節 因爲是從最後開始,一旦pos等於0也就是到了最前面,就無處可放了 因此拋出異常 不然,就在前面寫入一個 |
![image_5b98a4e3_d7b image_5b98a4e3_d7b](http://static.javashuo.com/static/loading.gif) 推回 指定字節數組b 從off偏移量開始的len個字節 推回len個長度,既然是從後往前設置數據,那麼pos的值就是可用 空間 的個數 若是空間不夠,拋出異常 不然,pos直接向前移動到指定位置 而後藉助於System.arraycopy 將b 便宜量off開始拷貝到 |
![image_5b98a4e3_1701 image_5b98a4e3_1701](http://static.javashuo.com/static/loading.gif) 是unread(byte[] b, int off, int len)的簡化形式 用於拷貝整個數組 |
從unread能夠清晰的看出來 內部的緩衝區,只是爲了給被回退掉的數據使用的 每次回退就是在最前面(下表索引較小的位置) 寫入數據 |
read
只要明確了unread以及內部緩衝區的存放形式便可很清楚的知道read的工做方式了 |
![image_5b98a4e3_5b63 image_5b98a4e3_5b63](http://static.javashuo.com/static/loading.gif) 既然是從最後開始往前依次存放,初始時 pos == buf.length 那麼但凡是有數據,pos必然,小於buf.length 也就是說,讀取一個字節時,若是回退緩衝內有數據,直接返回一個字節 不然,調用父類的read方法,而父類FilterInputStream 只是簡單的 封裝轉發請求 實際上就是直接使用內部的InputStream進行處理
|
多參數的read方法,將數據寫入到指定的字節數組b中
回退緩衝區中有效字節個數爲avail = buf.length - pos; 若是緩衝區中有數據,那麼將首先從回退緩衝區中讀取數據
若是讀取的字節數少於 len 個字節,那麼它將從底層輸入流中讀取字節
|
PushBackInputStream 不支持標記點相關的操做 |
|
總結
PushBackInputStream的實現邏輯比較清晰
我內部有一個緩衝區,你要是回退了我就往裏面寫入數據
每次的數據讀取都是先看看緩衝區裏面有沒有數據,有的話就先讀取回退緩衝區的
不然,就再去使用實際的流去進行讀取
看得出來,若是你歷來未曾回退過,那麼好像什麼都同樣,仍是使用原來的InputStream 進行讀取