国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - Java教程 - 通過JDK源碼學(xué)習(xí)InputStream詳解

通過JDK源碼學(xué)習(xí)InputStream詳解

2021-02-01 11:44汪洋之舟---seaboat Java教程

InputStream抽象類是所有字節(jié)輸入流的類的超類。這篇文章主要給大家介紹了關(guān)于通過JDK源碼學(xué)習(xí)InputStream的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編

概況

本文主要給大家介紹了通過JDK源碼學(xué)習(xí)InputStream的相關(guān)內(nèi)容,JDK 給我們提供了很多實用的輸入流 xxxInputStream,而 InputStream 是所有字節(jié)輸入流的抽象。包括 ByteArrayInputStream 、FilterInputStream 、BufferedInputStream 、DataInputStream 和 PushbackInputStream 等等。下面話不多說了,來一起看看詳細(xì)的介紹吧。

如何閱讀JDK源碼。

以看核心虛擬機(hotspot)code為例介紹。

1)熟悉虛擬機原理。調(diào)bug可以不懂原理,但是看code必須懂原理,從code里面看原理,基本不可能。hotspot的code寫的挺亂的,想直接通過code以及code中的注釋看明白還是很困難的。所以先熟悉虛擬機的原理,再去看code,會針對性比較強。

2)分模塊閱讀code。hotspot包括的模塊確實太多,我們需要分成不同的模塊各個擊破。以GC為例,hotspot中的gc算法有很多種,parallel scavenge,cms,g1…等等,先弄懂這些算法的原理,再去看code會比較快。不要看二手資料,不要看翻譯資料,推薦R大的hllvm論壇以及周志明的深入java虛擬機,hotspot源碼閱讀這本書寫的也還可以。

繼承結(jié)構(gòu)

?
1
2
--java.lang.Object
 --java.io.InputStream

類定義

?
1
public abstract class InputStream implements Closeable

InputStream 被定為 public 且 abstract 的類,實現(xiàn)了Closeable接口。

Closeable 接口表示 InputStream 可以被close,接口定義如下:

?
1
2
3
public interface Closeable extends AutoCloseable {
  public void close() throws IOException;
}

主要屬性

?
1
2
3
4
5
private static final int MAX_SKIP_BUFFER_SIZE = 2048;
 
private static final int DEFAULT_BUFFER_SIZE = 8192;
 
private static final int MAX_BUFFER_SIZE = Integer.MAX_VALUE - 8;
  • MAX_SKIP_BUFFER_SIZE 表示輸入流每次最多能跳過的字節(jié)數(shù)。
  • DEFAULT_BUFFER_SIZE 默認(rèn)的緩沖大小。
  • MAX_BUFFER_SIZE 表示最大的緩沖數(shù)組大小,這里設(shè)置為 Integer.MAX_VALUE - 8 這里也是考慮到 JVM 能支持的大小,超過這個值就會導(dǎo)致 OutOfMemoryError。

主要方法

read方法

一共有三個 read 方法,其中有一個抽象的 read 方法,其余兩個 read 方法都會調(diào)用這個抽象方法,該方法用于從輸入流讀取下一個字節(jié),返回一個0到255范圍的值。如果已經(jīng)到達(dá)輸入流結(jié)尾處而導(dǎo)致無可讀字節(jié)則返回-1,同時,此方法為阻塞方法,解除阻塞的條件:

     1. 有可讀的字節(jié)。

     2. 檢測到已經(jīng)是輸入流的結(jié)尾了。

     3. 拋出異常。

主要看第三個 read 方法即可,它傳入的三個參數(shù),byte數(shù)組、偏移量和數(shù)組長度。該方法主要是從輸入流中讀取指定長度的字節(jié)數(shù)據(jù)到字節(jié)數(shù)組中,需要注意的是這里只是嘗試去讀取長度為 len 的數(shù)組,但真正讀取到的數(shù)組長度不一定為 len,返回值才是真正讀取到的長度。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
public abstract int read() throws IOException;
 
public int read(byte b[]) throws IOException {
  return read(b, 0, b.length);
}
public int read(byte b[], int off, int len) throws IOException {
  if (b == null) {
    throw new NullPointerException();
  } else if (off < 0 || len < 0 || len > b.length - off) {
    throw new IndexOutOfBoundsException();
  } else if (len == 0) {
    return 0;
  }
 
  int c = read();
  if (c == -1) {
    return -1;
  }
  b[off] = (byte)c;
 
  int i = 1;
  try {
    for (; i < len ; i++) {
      c = read();
      if (c == -1) {
        break;
      }
      b[off + i] = (byte)c;
    }
  } catch (IOException ee) {
  }
  return i;
}

看看它的邏輯,數(shù)組為null則拋空指針,偏移量和長度超過邊界也拋異常,長度為0則什么都不敢直接返回0。接著調(diào)用 read() 讀取一個字節(jié),如果為-1則說明結(jié)束,直接返回-1。否則繼續(xù)根據(jù)數(shù)組長度循環(huán)調(diào)用 read() 方法讀取字節(jié),并且填充到傳入的數(shù)組對象中,最后返回讀取的字節(jié)數(shù)。

readAllBytes方法

該方法從輸入流讀取所有剩余的字節(jié),在此過程是阻塞的,直到所有剩余字節(jié)都被讀取或到達(dá)流的結(jié)尾或發(fā)生異常。

邏輯是用一個 for 循環(huán)內(nèi)嵌一個 while 循環(huán),while 循環(huán)不斷調(diào)用 read 方法嘗試將 DEFAULT_BUFFER_SIZE 長度的字節(jié)數(shù)組填滿,一旦填滿則需要將數(shù)組容量擴容一倍,再將原字節(jié)數(shù)組復(fù)制到新數(shù)組中,然后再通過 while 循環(huán)繼續(xù)讀取,直到達(dá)到尾部才跳出 for 循環(huán),最后返回讀取到的所有字節(jié)數(shù)組。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public byte[] readAllBytes() throws IOException {
  byte[] buf = new byte[DEFAULT_BUFFER_SIZE];
  int capacity = buf.length;
  int nread = 0;
  int n;
  for (;;) {
    while ((n = read(buf, nread, capacity - nread)) > 0)
      nread += n;
    if (n < 0)
      break;
    if (capacity <= MAX_BUFFER_SIZE - capacity) {
      capacity = capacity << 1;
    } else {
      if (capacity == MAX_BUFFER_SIZE)
        throw new OutOfMemoryError("Required array size too large");
      capacity = MAX_BUFFER_SIZE;
    }
    buf = Arrays.copyOf(buf, capacity);
  }
  return (capacity == nread) ? buf : Arrays.copyOf(buf, nread);
}

readNBytes方法

從輸入流中讀取指定長度的字節(jié),而且它能保證一定能讀取到指定的長度,它屬于阻塞方式,用一個 while 循環(huán)不斷調(diào)用 read 讀取字節(jié),直到讀取到指定長度才結(jié)束讀取。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
public int readNBytes(byte[] b, int off, int len) throws IOException {
  Objects.requireNonNull(b);
  if (off < 0 || len < 0 || len > b.length - off)
    throw new IndexOutOfBoundsException();
  int n = 0;
  while (n < len) {
    int count = read(b, off + n, len - n);
    if (count < 0)
      break;
    n += count;
  }
  return n;
}

available方法

返回從該輸入流能進(jìn)行非阻塞讀取的剩余字節(jié)數(shù),當(dāng)調(diào)用 read 讀取的字節(jié)數(shù)一般會小于該值,有一些InputStream的子實現(xiàn)類會通過該方法返回流的剩余總字節(jié)數(shù),但有些并不會,所以使用時要注意點。

這里抽象類直接返回0,子類中重寫該方法。

?
1
2
3
public int available() throws IOException {
    return 0;
  }

skip方法

從輸入流中跳過指定個數(shù)字節(jié),返回值為真正跳過的個數(shù)。這里的實現(xiàn)是簡單通過不斷調(diào)用 read 方法來實現(xiàn)跳過邏輯,但這是較低效的,子類可用更高效的方式重寫此方法。

下面看看邏輯,最大的跳過長度不能超過 MAX_SKIP_BUFFER_SIZE ,并且用一個 while 循環(huán)調(diào)用 read 方法,如果遇到返回為-1,即已經(jīng)到達(dá)結(jié)尾了,則跳出循環(huán)??梢钥吹?skipBuffer 其實是沒有什么作用,直接讓其被 GC 即可,最后返回真正跳過的字節(jié)數(shù)。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public long skip(long n) throws IOException {
 
  long remaining = n;
  int nr;
 
  if (n <= 0) {
    return 0;
  }
 
  int size = (int)Math.min(MAX_SKIP_BUFFER_SIZE, remaining);
  byte[] skipBuffer = new byte[size];
  while (remaining > 0) {
    nr = read(skipBuffer, 0, (int)Math.min(size, remaining));
    if (nr < 0) {
      break;
    }
    remaining -= nr;
  }
 
  return n - remaining;
}

close方法

此方法用于關(guān)閉輸入流,并且釋放相關(guān)資源 。

?
1
public void close() throws IOException {}

transferTo方法

從輸入流中按順序讀取全部字節(jié)并且寫入到指定的輸出流中,返回值為轉(zhuǎn)移的字節(jié)數(shù)。轉(zhuǎn)移過程中可能會發(fā)生不確定次的阻塞,阻塞可能發(fā)生在 read 操作或 write 操作。

主要邏輯是用 while 循環(huán)不斷調(diào)用 read 方法操作讀取字節(jié),然后調(diào)用輸出流的 write 方法寫入,直到讀取返回-1,即達(dá)到結(jié)尾。最后返回轉(zhuǎn)移的字節(jié)數(shù)。

?
1
2
3
4
5
6
7
8
9
10
11
public long transferTo(OutputStream out) throws IOException {
  Objects.requireNonNull(out, "out");
  long transferred = 0;
  byte[] buffer = new byte[DEFAULT_BUFFER_SIZE];
  int read;
  while ((read = this.read(buffer, 0, DEFAULT_BUFFER_SIZE)) >= 0) {
    out.write(buffer, 0, read);
    transferred += read;
  }
  return transferred;
}

markSupported方法

是否支持 mark 和 reset 操作,這里直接返回 false,子類根據(jù)實際重寫該方法。

?
1
2
3
public boolean markSupported() {
  return false;
}

mark方法

標(biāo)記輸入流當(dāng)前位置,與之對應(yīng)的是 reset 方法,通過他們之間的組合能實現(xiàn)重復(fù)讀取操作。另外它會傳入 readlimit 參數(shù),它用于表示該輸入流中在執(zhí)行 mark 操作后最多可以讀 readlimit 個字節(jié)后才使 mark 的位置失效。

可以看到 InputStream 的 mark 方法是什么都不做的,子類中再具體實現(xiàn)。

?
1
public synchronized void mark(int readlimit) {}

reset方法

與 mark 方法對應(yīng),它可以重置輸入流的位置到上次被 mark 操作標(biāo)識的位置。InputStream 的 reset 方法直接拋出一個 IOException,子類中根據(jù)實際情況實現(xiàn)。

?
1
2
3
public synchronized void reset() throws IOException {
  throw new IOException("mark/reset not supported");
}

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對服務(wù)器之家的支持。

原文鏈接:http://blog.csdn.net/wangyangzhizhou/article/details/78465531

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 国产精品一二三区视频出来一 | 国产1页| 久久性精品 | 97久久精品人人做人人爽50路 | 国产精品精品视频一区二区三区 | 亚洲精品国产综合区久久久久久久 | 在线精品日韩 | 福利成人 | 日韩精品一区二区三区四区五区 | 亚洲精品乱码久久久久久久久 | 久久综合久色欧美综合狠狠 | 二区视频 | 亚洲视频精品在线观看 | 欧美做爰一区二区三区 | 亚洲成人伦理 | 91成人短视频在线观看 | 黄色综合 | 亚洲精品国产乱码在线看蜜月 | 污污视频网站免费 | 免费大片黄在线观看 | 99久久国产露脸国语对白 | 岛国a视频 | 久久av综合 | 91精品国产乱码久久久久久 | 国产精品免费视频观看 | 亚洲成人一区二区三区四区 | 日本中文字幕在线播放 | 日韩在线小视频 | 国产美女久久久 | 亚洲视频中文字幕 | 中文字幕在线观看一区二区三区 | 久久精品一区二区国产 | 99久久成人 | 欧美午夜精品久久久久久浪潮 | 国产97色在线 | 亚洲 | 成人精品视频99在线观看免费 | 亚洲视频在线免费观看 | 自拍视频在线 | 亚州国产 | 成人在线免费小视频 | 日韩精品1区 |