• java I/O框架 (三)基本流


    概述

      基本流有字节输入输出流(InputStream,OutputStream),和字符输入输出流(Reader,Writer),它们都是抽象类,作为Java IO API中其他所有流的父类存在.

      我们知道数据存储在介质中时是以二进制的形式存在,用字节(byte)形式就可以实现读写,在jdk1.0版本中也只有字节输入输出流(InpuStream,OutputStream),在之后的1.1版本才加入了字符输入输出流(Reader,Writer),这是为了满足一些文本读写方便,比如中文字符可能占有两到三个字节,这时如果使用字节流,则不便使用,可能造成乱码,而字符流则可以根据字符编码一次读写一个字符.

      在开始学习这四种基本流时,将列出它所有的成员方法,一一说明,基本流的成员方法比较少,而且他的各种功能的子类都是通过重写这些方法来实现不同功能,在后面学习具体类时将不再这样列举

    InputStream

      InputStream是java IO所有字节输入流的超类,如果你自己想要实现一个字节输入流组件,最好直接继承InputStream而不是他的子类,这样你可以处理所有类型的输入流.从上图中我们可以看到InputStream实现了Closeable接口,实际四个基本流都实现了此接口,其接口close()方法作为流操作结束时释放系统资源用.下面我们着重看InputStream的成员方法.

    public abstract int read() throws IOException;

         read()是一个抽象方法,由子类重写具体实现方式,它用来读取数据,且每次只读取一个字节,并以int类型返回该字节,当InputStream读取到文件末尾,再也没有数据时,则返回-1;

    public int read(byte b[]) throws IOException{...}

      read(byte[])方法一次读取多个字节,并将其储存在传入的byte数组中,相比read(),其速度提高了很多.read(byte[])返回的是int类型所读取字节数,和read()一样,到文件末尾时,返回-1.

    public int read(byte[] b,int off,int len) throws IOException{...}

      read(byte[],int,int)  read(byte[])类似,不同的是read(byte[] b ,int off,int len)可以通过传入参数off选择开始要写入到的数组索引位置,以及len确定要读取的字节数,具体细节可以看源码

    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) {   //这里要判断传入的参数是否正常  其中 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;  //将读入的第一个字节放入选择的数组索引off处
     
             int i = 1;     //变量i用来计实际读入数组的字节数
             try {
                 for (; i < len ; i++) {
                     c = read();
                     if (c == -1) {
                         break;
                     }
                     b[off + i] = (byte)c;
                 }
             } catch (IOException ee) {
             }
             return i;
         }
    View Code 

    public long skip(long n) throws IOException{...}

      skip(long)用来跳过不读一些字节数,InputStream定义了一个MAX_SKIP_BUFFER_SIZE常量,大小为2048,这是一次所能跳过的最大字节数,其返回值为long类型的实际跳过字节数;

    public int available() throws IOException{...}

      available()用来获取流中可读取数据大小,InputStream中此方法默认返回0,由其子类重写来实现具体方法;

    public void close() throws IOException{}

      close()方法用来读取结束后,关闭流,释放系统资源,此方法一定要确保执行,否则会大量系统资源被占用得不到释放,会导致内存溢出.

    public boolean markSupported(){}

      markSupported()是一个标记方法,用来表示是否支持mark(int)和reset()方法,InputStream 默认返回false,如果要想支持mark(int)和reset()方法,应当重写markSupport()使其返回true.

    public synchronized void mark(int readlimt) {}

      mark(int)方法用来标记一个读取的文件字节位置,当执行reset()方法时,InputStream可以重新从此处读取,这适用于一些分析器,这些分析器有时需要预读后面一部分字节,如果没有发现目标,则返回重写读取.

    public synchronized void reset() throws IOException{}

      reset()方法将流读取重置为mark(int)方法标记的位置

    OutputStream

       OutputStream是java IO所有字节输出流的超类,它实现了两个接口,Closeable和Flushable,其中Flushable接口的flush()方法是将缓冲区的数据直接刷入到输入流所要写入的对象,具体将在讲缓冲流的时候分析.下面我们看OutputStream的成员方法;

      public abstract void write(int b) throws IOException;
        write(int)方法用来写入一个字节的数据,但方法里传入的是一个int类型数据,不过这里只会写入int类型数据第一个字节,后面三个字节将被抛弃,也就是写入低8位,忽略高24位

      public void write(byte b[]) throws IOException
        write(byte[])方法是将一个数组的数据写入到字节输出流中,

      public void write(byte b[], int off, int len) throws IOException
        和write(byte[])一样的用法,不同的是这里可以从数组指定索引处开始写入,并且可以设定要写入的字节长度,细节可以查看源码
        public void write(byte b[], int off, int len) throws IOException {
            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]);
            }
        }
    View Code
    
    
      public void flush() throws IOException {}
         在使用缓冲流时,写入到流的数据并不会立即写入磁盘中,而是存在一个缓冲区,等到缓冲区满后才会执行实际写入,如果调用flush()方法,则立即执行将缓冲区数据写入到文件,flush()方法在OutputStream中方法体为空,需要子类重写去实现.
      

      public void close() throws IOException {}      
         同InputStream

    Reader

    
    

    Reader是java io中所有字符输入流的超类,它的大部分方法和InputStream相同,只不过Reader是以字符的形式读取数据,所以在一些需要人们阅读的文件读取上使用Reader.Reader相比InputStream多实现了一个Readerable接口,它使Reader可以将读取的数据存储在一个缓冲区内.而且Reader有两个构造方法,可以使用自己或者传入的对象作为同步锁,以保持读取同步,下面和InputStream相同的方法不再列举,可以自己参考InputStream.

    public int read(java.nio.CharBuffer target) throws IOException {...}
    public int read() throws IOException {...}

      一次读取一个字符,并将该字符以int类型返回,当读到文件尾没有数据时,返回-1
    public int read(char cbuf[]) throws IOException {...}

      一次读取多个字符,并将字符放到传入的字符数组中,返回值为实际读取的字符数,到文件尾没有数据时返回-1
    abstract public int read(char cbuf[], int off, int len) throws IOException;

      一次读取多个字符,并将字符放入传入的字符数组中,并且可以指定传入字符数组索引和传入的字符数,返回值同样为实际读取字符数,到文件尾没有数据时返回-1

    public boolean ready() throws IOException {...}

      ready()告诉程序字符流是否可以被读取,Reader中默认返回false

    Writer

     Writer是java IO中所有字符输出流的超类,它的大部分方法和OutputStream一样,不同的是Writer写入的是字符.同Reader一样Writer有两个构造方法,可以使用自己或者传入的对象作为同步锁,以保证写入同步,,此外Writer还实现了一个Appenderable接口,可以追加写入字符

     
  • 相关阅读:
    mysql 复制表数据,表结构的3种方法
    MySQL 存储过程使用表名做参数
    关于mysql engine(引擎)的疑问
    mysql存储过程之循环
    mysql 命令大全
    关于mysql的表名/字段名/字段值是否区分大小写的问题
    navicat for mysql 快捷键(原创)
    解决"Subquery returns more than 1 row"sql查询错误
    mysql:“Access denied for user 'root@IP地址'"
    MySQL常用经典语句
  • 原文地址:https://www.cnblogs.com/bushi/p/6554799.html
Copyright © 2020-2023  润新知