在看《《Java程序性能优化-让你的Java程序更快、更稳定》(葛一宁等编著)》一书时,文中提到JDK中OutputStream和InputStream使用观察者模式,提高I/O性能。
如下代码:

1
DataOutputStream dos = new DataOutputStream(new BufferedOutputStream(new FileOutputStream("d:test.txt")));

通过结合BufferedOutputStream对数据做了缓存处理,调用write方法时并不是每次都会直接写入磁盘。只有当缓冲区满的时候,才会写入磁盘。
代码片段:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public synchronized void write(byte b[], int off, int len) throws IOException {
    if (len >= buf.length) {
        /* If the request length exceeds the size of the output buffer,
           flush the output buffer and then write the data directly.
           In this way buffered streams will cascade harmlessly. */
        flushBuffer();
        out.write(b, off, len);
        return;
    }
    if (len > buf.length - count) {
        flushBuffer();
    }
    System.arraycopy(b, off, buf, count, len);
    count += len;
}

如上代码,最后实际是调用的FileOutputStream的write方法写入磁盘的。最终调用的是FileOutputStream的writeBytes方法,这是一个native方法。

1
2
private native void writeBytes(byte b[], int off, int len, boolean append)
        throws IOException;

可以看到,它的实现并不在FileOutputStream。

那么,到底最终是如何写入磁盘的呢?

native关键字说明其修饰的方法是一个原生态方法,方法对应的实现不是在当前文件,而是在用其他语言(如C和C++)实现的文件中。Java语言本身不能对操作系统底层进行访问和操作,但是可以通过JNI接口调用其他语言来实现对底层的访问。

具体可以参考这2篇文章:Java中native关键字在 Windows 中实现 Java 本地方法.