Java

Java IO笔记(InputStream/OutputSter

2019-12-16  本文已影响0人  moonfish1994

(最近刚来到简书平台,以前在CSDN上写的一些东西,也在逐渐的移到这儿来,有些篇幅是很早的时候写下的,因此可能会看到一些内容杂乱的文章,对此深感抱歉,以下为正文)


正文

本篇讲述的内容是Java IO中的InputStream类和OutputStream类。

我们知道流是一种抽象的概念,抽象了输入输出设备。在Java中,所有的流对象都位于java.io包中,其中有4个流是特殊的,通过它们可以衍生出所有的流对象,它们分别是InputStream/OutputStream和Reader/Writer。它们是通过流中数据的单位来区分的。本次讲述的是其中的字节流,InputStream和OutputStream。

这四个基类都是抽象类,其中定义了一些流所具备的方法,下面将贴出InputStream和OutputStream的源码,来进行分析:

InputStream.java

package java.io;
 
public abstract class InputStream implements Closeable {
 
    //声明了一个最终常量,MAX_SKIP_BUFFER_SIZE,该常量限制了下面skip方法中所创建的buffer缓冲区的最大容量。
    private static final int MAX_SKIP_BUFFER_SIZE = 2048;
 
    /**
     *从流中读取一个byte的数据,并返回一个int值,如果到达了流的尾端或者没有数据可读,将返回-1。
     */
    public abstract int read() throws IOException;
 
    /**
     *一个带参的read方法,传入一个byte数组,内部调用带3个参数的read方法,用整个数组去读取数据。
     */
    public int read(byte b[]) throws IOException {
        return read(b, 0, b.length);
    }
 
    /**
     * 一个带3个参数的read方法,第一个参数是存放数据的数组,第二个参数是开始存放数据的起点,第三个参数是数据存储的长度。
     *返回一个int值,该值为实际成功读取的byte数据的次数。如过没有成功读取到有用的数据,则返回-1。
     */
    public int read(byte b[], int off, int len) throws IOException {
    //对传入参数的安全监测。如果传入数组为空报空指针异常,如果读取起点、读取长度小于0或者读取起点加上读取长度大于数组本身长度抛越界异常,如果读取长度
    //为0,则返回0,表示读取了0个数据
        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;
 
    //通过循环读取,为byte数组赋值,同时每次循环,使得i+1,最终返回i的值。
        int i = 1;
        try {
            for (; i < len ; i++) {
                c = read();
                if (c == -1) {
                    break;
                }
                b[off + i] = (byte)c;
            }
        } catch (IOException ee) {
        }
        return i;
    }
 
    /**
     * 一个带参的skip方法,参数为一个long型数据,表示跳跃读取的长度,返回一个long型数据,表示实际已经跳过的数据长度。
     */
    public long skip(long n) throws IOException {
 
        long remaining = n;
        int nr;
 
        if (n <= 0) {
            return 0;
        }
 
    //建立一个最大容量为2048的byte数组,内部调用带3个参数的read方法来起到跳过的作用。每读取一次,都将remaining减去读取过的值,最终通过n-r来得到实际跳
    //过的数据长度。
        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;
    }
 
    /**
     * 来获取流中实际可以读取的数据的总长度,该方法应该被子类重写,应为InputStream中该方法的返回值永远为0。当然该方法的返回值不一定是准确的,有兴趣的朋友
     *可以自行查阅相关资料,java源码中给出的官方注释也解释的蛮清楚的,此处就不再多说了。
     */
    public int available() throws IOException {
        return 0;
    }
 
    /**
     * 该方法用于关闭流
     */
    public void close() throws IOException {}
 
    /**
     *该方法用于标记流当前的读取位置
     */
    public synchronized void mark(int readlimit) {}
 
    /**
     *该方法用于将流的读取位置返回至标记处
     */
    public synchronized void reset() throws IOException {
        throw new IOException("mark/reset not supported");
    }
 
    /**
     *该方法用于返回当前流是否支持标记功能,InputStream永远返回为false。
     */
    public boolean markSupported() {
        return false;
    }
 
}

OutputStream.java

package java.io;
 
public abstract class OutputStream implements Closeable, Flushable {
    /**
     * 一个带参的write方法,传入的参数为一个int型的值,该方法每次写出一个byte数据,传入的int型数据b为要写出的数据。
     */
    public abstract void write(int b) throws IOException;
 
    /**
     *一个带参的write方法,传入的参数为一个byte型数组,该方法将数组中的所有数据写出。
     */
    public void write(byte b[]) throws IOException {
        write(b, 0, b.length);
    }
 
    /**
     * 一个带三个参数的write方法,第一个参数为一个byte型数组,其中包含了写出数据,第二个参数为写出的起始位置,第三个参数为写出数据的长度。
     */
    public void write(byte b[], int off, int len) throws IOException {
        //对传入参数的安全监测。如果传入数组为空报空指针异常,如果写出起点、写出长度小于0或者写出起点大于数组本身长度则抛越界异常,如果写出长度
    //为0,则返回0,表示写出了0个数据
 
        if (b == null) {
            throw new NullPointerException();
        } else if ((off < 0) || (off > b.length) || (len < 0) ||
                   ((off + len) > b.length) || ((off + len) < 0)) {
            throw new IndexOutOfBoundsException();
        } else if (len == 0) {
            return;
        }
        for (int i = 0 ; i < len ; i++) {
            write(b[off + i]);
        }
    }
 
    /**
     * 该方法用于将缓存区中的数据写入目的地。
     */
    public void flush() throws IOException {
    }
 
    /**
     * 该方法用于关闭当前流
     */
    public void close() throws IOException {
    }
 
}

以上为本篇的全部内容。

上一篇 下一篇

猜你喜欢

热点阅读