Java IO入门(含源码解析)

Java IO

Java的输入输出流,用于和程序外部交换数据。Java的IO是通过流式传输的

流的链接机制

可将一个流和另一个流首尾相接,以将输入数据转换成相应的输出数据

IO流接口

Java的IO流接口

对应抽象类
输入流 InputStream/Reader
输出流 OutputStream/Writer
字节流 InputStream/OutputStream
字符流 Reader/Writer

InputStream

字节输入流,以字节为单位从程序外部获取数据
API

类型 作用 API
统计 流内有效字节数 available()
读出一个字节 read()
读出一定长度字节 read(byte[],int offset,lenght)
跳过 跳过一定长度字节开始读 skip(long n)
回溯 能否标记 markSupported
标记当前读的位置 mark(int readlimit)
回到标记开始读 reset
关闭 清空缓冲区 flush()
关闭流 close()

InputStream继承关系类图

InputStream继承类图

真是令人望而生畏!

InputStream节选继承关系类图

InputStream继承类图-节选

节选了几个重要输入流实现类

从类图可以清楚地看到,左边三个实现类有InputStream的引用,意味着他们内部有流的链接。而右侧几个实现类则最多只能链接同类型的输入流

实现类 描述
ObjectInputSteam 从其他InputStream中读出Object
SequenceInputStream 持有一组InputStream,读完一个换一个
FilterInputStream 从其他InputStream过滤输入数据并读出
StringInptStream 从字符串或其他String输入流中读出字节数据
ByteArrayInputStream 从字节数组中读出字节数据
PipedInputStream
FileInputStream 从文件中读字节数据

实现类源码解析

简略解析,其中ObjectInputStream和PipedInputStream由于篇幅较长,以后再解析

ByteArrayInputStream

从字节数组中读出字节数据
成员变量

成员变量 含义
byte buf[] 流的数据缓冲
int pos 当前读的位置
int mark = 0 标记的位置
int count 缓冲长度

构造函数

//构造函数提供字节数组源,直接复制到流的缓冲区,初始化下一次读的位置以及缓冲区大小
public ByteArrayInputStream(byte buf[]) {
    this.buf = buf;
    this.pos = 0;
    this.count = buf.length;
}

读数据

//缓冲无可用数据返回-1,有就读一个字节放在int的后8位返回并更新下一次读的位置
public synchronized int read() {
    return (pos < count) ? (buf[pos++] & 0xff) : -1;
}
//缓冲无可用数据返回-1,否则复制到参数提供的数组中,返回实际复制的字节数
public synchronized int read(byte b[], int off, int len) {
    if (b == null) {
        throw new NullPointerException();
    } else if (off < 0 || len < 0 || len > b.length - off) {
        throw new IndexOutOfBoundsException();
    }

    if (pos >= count) {
        return -1;
    }

    int avail = count - pos;
    if (len > avail) {
        len = avail;
    }
    if (len <= 0) {
        return 0;
    }
    System.arraycopy(buf, pos, b, off, len);
    pos += len;
    return len;
}

跳着读

//下次读的位置+n,如果超过缓冲区尾部则设为缓冲区尾部
public synchronized long skip(long n) {
    long k = count - pos;
    if (n < k) {
        k = n < 0 ? 0 : n;
    }

    pos += k;
    return k;
}

回溯读

//支持mark和reset
public boolean markSupported() {
    return true;
}
//设置标记为下一次要读的位置
public void mark(int readAheadLimit) {
    mark = pos;
}
//设置下一次要读的位置为标记
public synchronized void reset() {
    pos = mark;
}
StringInputStream

从字符串或其他String输入流中读出字节数据,继承自ReaderInputStream,持有该一个Reader对象用于读取字符串,然后拆分字符为字节数据读出

//字符输入流
private Reader in;
//编码
private String encoding;
//字符中读剩下的字节的缓冲
private byte[] slack;
//缓冲起始位
private int begin;

public synchronized int read() throws IOException {
    if (this.in == null) {
        throw new IOException("Stream Closed");
    } else {
        byte result;
        //有上次读剩的缓冲,从缓冲读
        if (this.slack != null && this.begin < this.slack.length) {
            result = this.slack[this.begin];
            if (++this.begin == this.slack.length) {
                this.slack = null;
            }
            //无缓冲,从字符输入流读一个字符到缓冲,返回缓冲第一位
        } else {
            byte[] buf = new byte[1];
            if (this.read(buf, 0, 1) <= 0) {
                return -1;
            }

            result = buf[0];
        }

        return result & 255;
    }
}
FileInputStream

文件输入流,从文件中读取字节数据

public int read() throws IOException {
    return read0();
}
//需要本地方法读取文件
private native int read0() throws IOException;
SequenceInputStream

序列字节输入流,持有一组字节输入流,依次进行读出,读完一个切换一个
成员变量

成员变量 含义
Enumeration<? extends InputStream> e 持有的字节输入流
InputStream in 当前读的字节输入流

read实现

public int read() throws IOException {
    while (in != null) {
        //读当前InputStream,读完了切换
        int c = in.read();
        if (c != -1) {
            return c;
        }
        nextStream();
    }
    return -1;
}

final void nextStream() throws IOException {
    //关闭现在的输入流
    if (in != null) {
        in.close();
    }
    //切换下一个输入流
    if (e.hasMoreElements()) {
        in = (InputStream) e.nextElement();
        if (in == null)
            throw new NullPointerException();
    }
    else in = null;

}
FilterInputStream

FilterInputStream使用代理模式,封装了一个字节输入流对象,在重写的方法中通过调用字节输入流对象的方法读取数据,再通过过滤代码(由子类重写read方法添加)实现对输入数据的过滤

//持有的字节输入流对象
protected volatile InputStream in;
//构造函数
protected FilterInputStream(InputStream in) {
    this.in = in;
}
//读取一个字节
public int read() throws IOException {
    return in.read();
}

OutputStream

字节输出流,以字节为单位输出
API

类型 作用 API
写入一个字节 write(int)
写入一个字节数组 read/write(byte[],int offset,lenght)
关闭 清空缓冲区 flush()
关闭流 close()

Reader

字符输入流,以字符为单位从程序外部获取数据
API

类型 作用 API
读出一个字符读出一个字节 read()
读出一定长度字符 read/write(char[],offset,len)
跳过 跳过一定长度字节开始读 skip(long n)
回溯 能否标记 markSupported
标记当前读的位置 mark(int readlimit)
回到标记开始读 reset
关闭 清空缓冲区 flush()
关闭流 close()

Writer

API

类型 作用 API
写入一定长度字符 append/write(char[],offset,len)
写入一个字符串 append/write(String s)
写入一个字符串的指定区间 append/write(String s,offset,len)
关闭 关闭流 close
清空流内缓冲 flush()
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容