[十三]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 進行讀取






免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM