之前看過相關的ByteBuffer的使用,但是問題是那時還年輕,所以現在有點老了,因此,忘記了,所以決心看源代碼了解一番----故作此篇文章。
?
查看ByteBuffer的API,看的我是一頭霧水,搞不清什么mark、position、limit、flip、reset幾個的用法,先看下面的例子:
- ?????String?str?=? "helloWorld" ;??
- ????????ByteBuffer?buff??=?ByteBuffer.wrap(str.getBytes());??
- ????????System.out.println( "position:" +buff.position()+ "\t?limit:" +buff.limit());??
- ???????? //讀取兩個字節 ??
- ????????buff.get();??
- ????????buff.get();??
- ????????System.out.println( "position:" +?buff.get(buff.position())+ "\t?limit:" +buff.limit());??
- ????????buff.mark();??
- ????????System.out.println( "position:" +buff.position()+ "\t?limit:" +buff.limit());??
- ????????buff.flip();??
- ????????System.out.println( "position:" +buff.position()+ "\t?limit:" +buff.limit());<span?style= "white-space:?pre;" >????</span>??
- <pre?name= "code" ? class = "java" >輸出結果:??
- position: 0 ??limit: 10 ??
- position: 2 ??limit: 10 ??
- position: 2 ??limit: 10 ??
- position: 0 ??limit: 2 ??
- </pre>??
- ???
??我們以每位開發人員熟悉的”helloworld“,用ByteBuffer將字符串包裝,由于ByteBuffer是一個抽象類,通過wrap包裝的對象將實際返回的是一個HeapByteBuffer對象。由此可知HeapByteBuffer是ByteBuffer的子類,同樣的ByteBuffer又是Buffer抽象類的子類。以上提到的mark、position、limit、flip、reset都是出自于Buffer這個抽象類。
下面我們來解析幾個方法的,當我們調用了wrap方法后Buffer中初始化的結構是:
注釋:
m:mark;
p:position;
L:limit;
?
???初始情況下mark是指向第一個元素之前的的即-1,postion為指向第一個元素為0.而Limit是被賦值為byte[]的長度。
因此這就是打印結果的第一行。
m | ? | ? | p | ? | ? | ? | ? | ? | ? | ? | L |
-1 | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | ? |
? | H | E | L | L | O | W | O | R | L | D | ? |
?? 當我們連續調用兩次get()方法獲得兩個個字節,每次調用都會觸發position++操作,那么此時position就會移動到index = 2的的地方,而這個時候Limit和mark是不會發生變化的。如果將讀取的兩個字節打印會是H和E,因此執行結果第二行會有
position:2? limit:10結果.
? | ? | ? | m、p | ? | ? | ? | ? | ? | ? | ? | L |
-1 | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | ? |
? | H | E | L | L | O | W | O | R | L | D | ? |
???? 讀取完畢后我們使用mark,這個時候mark會從-1移動到2和position指向同一個元素,可以看見Limit是不會發生改變的。
m | p | ? | L | ? | ? | ? | ? | ? | ? | ? | ? |
-1 | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | ? |
? | H | E | L | L | O | W | O | R | L | D | ? |
??? 使用了mark標記的當前的position后,如果們調用flip,這個時候Limit就會指向position的位置,并將mark和position還原為初始值。這樣就知道了limit當前的就為2,什么意思呢?就是說當前可以讀的字節數是2。
我們可以嘗試一下如下代碼:
?
?? ?輸出結果:he
- System.out.println(( char )buff.get()+ "" +( char )buff.get());??
??? 貌似這也沒什么稀奇的,如果你在代碼換成
?
?? 為什么會拋異常呢?原因是limit的含義就想一個窗口,你當前能讀到的數據就是當前窗口限制的(本例中即為2),如果這個窗口之外的所有元素都是不可讀的。至此我想你和我就應該明白這幾個參數的含義了吧。
- //?System.out.println((char)buff.get()+""+(char)buff.get() ??
- ???System.out.println(( char )buff.get()+ "" +( char )buff.get()+ "" +( char )buff.get());??
- 輸出結果:<pre?name= "code" ? class = "java" >position: 0 ????limit: 10 ??
- Exception?in?thread? "main" ?java.nio.BufferUnderflowException??
- ????at?java.nio.Buffer.nextGetIndex(Buffer.java: 474 )??
- ????at?java.nio.HeapByteBuffer.get(HeapByteBuffer.java: 117 )??
- ????at?com.taobao.moxing.notify.Main.main(Main.java: 33 )position: 2 ????limit: 10 ??
- position: 2 ???limit: 10 ??
- position: 0 ???limit: 2 ??
- </pre>??
??? 而至于reset方法,它是將當前的position設置為0,
???? rewind是將mark重置為-1,position重置為0;
???? clear方法是真正的重置,將mark=-1,position=0,limit=capacity(即當前buffer的容量)
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
