字节流

IO流

IO的分类:

根据数据的流向分为:输入流输出流

  • 输入流 :把数据从其他设备上读取到内存中的流。
  • 输出流 :把数据从内存 中写出到其他设备上的流。

格局数据的类型分为:字节流字符流

  • 字节流 :以字节为单位,读写数据的流。
  • 字符流 :以字符为单位,读写数据的流。
输入流 输出流
字节流 字节输入流
InputStream
字节输出流
OutputStream
字符流 字符输入流
Reader
字符输出流
Writer

字节流

一切皆为字节:

一切文件数据(文本、图片、视频等)在存储时,都是以二进制数字的形式保存,都一个一个的字节,那么传输时一样如此。所以,字节流可以传输任意文件数据。在操作流的时候,我们要时刻明确,无论使用什么样的流对象,底层传输的始终为二进制数据。

字节输出流OutputStream:

java.io.OutputStream 抽象类是表示字节输出流的所有类的超类,将指定的字节信息写出到目的地。它定义了字节输出流的基本共性功能方法。

  • public void close() :关闭此输出流并释放与此流相关联的任何系统资源。close方法,当完成流的操作时,必须调用此方法,释放系统资源。
  • public void flush() :刷新此输出流并强制任何缓冲的输出字节被写出。
  • public void write(byte[] b):将 b.length字节从指定的字节数组写入此输出流。
  • public void write(byte[] b, int off, int len) :从指定的字节数组写入 len字节,从偏移量 off开始输出到此输出流。
  • public abstract void write(int b) :将指定的字节输出流。

    FileOutputStream类:

OutputStream有很多子类,我们从最简单的一个子类开始。

java.io.FileOutputStream 类是文件输出流,用于将数据写出到文件。

构造方法:

​ FileOutputStream(String name)创建一个向具有指定名称的文件中写入数据的输出文件流。
​ FileOutputStream(File file) 创建一个向指定 File 对象表示的文件中写入数据的文件输出流。
​ 参数:写入数据的目的
​ String name:目的地是一个文件的路径
​ File file:目的地是一个文件
​ 构造方法的作用:
​ 1.创建一个FileOutputStream对象
​ 2.会根据构造方法中传递的文件/文件路径,创建一个空的文件
​ 3.会把FileOutputStream对象指向创建好的文件

写入数据的原理(内存-->硬盘):
java程序-->JVM(java虚拟机)-->OS(操作系统)-->OS调用写数据的方法-->把数据写入到文件中

字节输出流的使用步骤(重点):
1.创建一个FileOutputStream对象,构造方法中传递写入数据的目的地
2.调用FileOutputStream对象中的方法write,把数据写入到文件中
3.释放资源(流使用会占用一定的内存,使用完毕要把内存清空,提供程序的效率)

//往指定路径写入一个a,一个字节
package InputOutput;
import java.io.FileOutputStream;
import java.io.IOException;
public class Demo1 {
    public static void main(String[] args) throws IOException {
        FileOutputStream fos=new FileOutputStream("E:\\美图\\a.txt");
        fos.write(97);
        fos.close();
    }
}

file

file

文件存储的原理:

file

一次写多个字节:

public void write(byte[] b):将 b.length字节从指定的字节数组写入此输出流。
public void write(byte[] b, int off, int len) :从指定的字节数组写入 len字节,从偏移量 off开始输出到此输出流。

/*
            public void write(byte[] b):将 b.length字节从指定的字节数组写入此输出流。
            一次写多个字节:
                如果写的第一个字节是正数(0-127),那么显示的时候会查询ASCII表
                如果写的第一个字节是负数,那第一个字节会和第二个字节,两个字节组成一个中文显示,查询系统默认码表(GBK)
         */
package InputOutput;
import java.io.FileOutputStream;
import java.io.IOException;
public class Demo1 {
    public static void main(String[] args) throws IOException {
        /*FileOutputStream fos=new FileOutputStream("E:\\美图\\a.txt");*/
        /*FileOutputStream fos=new FileOutputStream("E:\\美图\\b.txt");*/
        FileOutputStream fos=new FileOutputStream("E:\\美图\\c.txt");
       /* byte [] bytes1={65,66,67,68,69};*/
        byte [] bytes2={-65,66,-67,68,69};
       /* fos.write(bytes1);*/
        fos.write(bytes2);
        fos.close();
    }
}

//可以直接打印字符串,只要调用字符串中的getBytes()方法即可
package InputOutput;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.Arrays;
public class Demo1 {
    public static void main(String[] args) throws IOException {
        FileOutputStream fos=new FileOutputStream("E:\\美图\\d.txt");
        byte[] bytes = "年轻肆无忌惮".getBytes();
        System.out.println(Arrays.toString(bytes));//[-27, -71, -76, -24, -67, -69, -24, -126, -122, -26, -105, -96, -27, -65, -116, -26, -125, -82]
        fos.write(bytes);
        fos.close();
    }
}

file

file

file

数据的追加写与换行写:

再往一个文件中写数据不会接着之前写的,而是会覆盖之前写的内容,这就是追加写要解决的问题

/*
    追加写/续写:使用两个参数的构造方法
        FileOutputStream(String name, boolean append)创建一个向具有指定 name 的文件中写入数据的输出文件流。
        FileOutputStream(File file, boolean append) 创建一个向指定 File 对象表示的文件中写入数据的文件输出流。
        参数:
           String name,File file:写入数据的目的地
           boolean append:追加写开关
            true:创建对象不会覆盖源文件,继续在文件的末尾追加写数据
            false:创建一个新文件,覆盖源文件
    写换行:写换行符号
        windows:\r\n
        linux:/n
        mac:/r
 */
package InputOutput;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.Arrays;
public class Demo1 {
    public static void main(String[] args) throws IOException {
        FileOutputStream fos=new FileOutputStream("E:\\美图\\d.txt",true);
        fos.write("\r\n".getBytes(StandardCharsets.UTF_8));
        byte[] bytes = "竟是我自己".getBytes();
        System.out.println(Arrays.toString(bytes));
        fos.write(bytes);
        fos.close();
    }
}

file

字节输入流:

java.io.InputStream 抽象类是表示字节输入流的所有类的超类,可以读取字节信息到内存中。它定义了字节输入流的基本共性功能方法。

  • public void close() :关闭此输入流并释放与此流相关联的任何系统资源。
  • public abstract int read(): 从输入流读取数据的下一个字节。
  • public int read(byte[] b): 从输入流中读取一些字节数,并将它们存储到字节数组 b中 。

小贴士:

close方法,当完成流的操作时,必须调用此方法,释放系统资源。

FileInputStream类

java.io.FileInputStream 类是文件输入流,从文件中读取字节。

构造方法:
FileInputStream(String name)
FileInputStream(File file)
参数:读取文件的数据源
String name:文件的路径
File file:文件
构造方法的作用:
1.会创建一个FileInputStream对象
2.会把FileInputStream对象指定构造方法中要读取的文件

读取数据的原理(硬盘-->内存):
java程序-->JVM-->OS-->OS读取数据的方法-->读取文件

字节输入流的使用步骤(重点):
1.创建FileInputStream对象,构造方法中绑定要读取的数据源
2.使用FileInputStream对象中的方法read,读取文件
3.释放资源

 //int read()读取文件中的一个字节并返回,读取到文件的末尾返回-1
package InputOutput;
import java.io.FileInputStream;
import java.io.IOException;
public class Demo02 {
    public static void main(String[] args) throws IOException {
        FileInputStream fis=new FileInputStream("E:\\美图\\a.txt");
        int len=0;
        /*布尔表达式(len = fis.read())!=-1
        1.fis.read():读取一个字节
        2.len = fis.read():把读取到的字节赋值给变量len
        3.(len = fis.read())!=-1:判断变量len是否不等于-1*/
        while((len=fis.read())!=-1) {
            System.out.println((char)len);
        }
    }
}

file

file

一次读取多个字节:

/*
    字节输入流一次读取多个字节的方法:
        int read(byte[] b) 从输入流中读取一定数量的字节,并将其存储在缓冲区数组 b 中。
    明确两件事情:
        1.方法的参数byte[]的作用?
            起到缓冲作用,存储每次读取到的多个字节
            数组的长度一把定义为1024(1kb)或者1024的整数倍
        2.方法的返回值int是什么?
            每次读取的有效字节个数

    String类的构造方法
        String(byte[] bytes) :把字节数组转换为字符串
        String(byte[] bytes, int offset, int length) 把字节数组的一部分转换为字符串 offset:数组的开始索引 length:转换的字节个数
 */
package InputOutput;
import java.io.FileInputStream;
import java.io.IOException;
public class Demo03 {
    public static void main(String[] args) throws IOException {
        FileInputStream fis=new FileInputStream("E:\\美图\\b.txt");
        byte [] bytes=new byte[2];
        int len=fis.read(bytes);
        System.out.println(len);
        System.out.println(new String(bytes));
    }
}

file

file

读取数据的原理:

file

字节流练习:图片复制

文件复制的原理:

file

图片复制:

/*文件复制的步骤:
        1.创建一个字节输入流对象,构造方法中绑定要读取的数据源
        2.创建一个字节输出流对象,构造方法中绑定要写入的目的地
        3.使用字节输入流对象中的方法read读取文件
        4.使用字节输出流中的方法write,把读取到的字节写入到目的地的文件中
        5.释放资源*/
package InputOutput;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
public class Demo04 {
    public static void main(String[] args) throws IOException {
        long s = System.currentTimeMillis();
        //1.创建一个字节输入流对象,构造方法中绑定要读取的数据源
        FileInputStream fis =new FileInputStream("E:\\美图\\新建文件夹\\1.jpg");
        //2.创建一个字节输出流对象,构造方法中绑定要写入的目的地
        FileOutputStream fos =new FileOutputStream("E:\\美图\\1.jpg");
        //一次读取一个字节写入一个字节的方式
        //3.使用字节输入流对象中的方法read读取文件
        int len = 0;//每次读取的有效字节个数
        while((len = fis.read())!=-1){
            //4.使用字节输出流中的方法write,把读取到的字节写入到目的地的文件中
            fos.write(len);
        }
        //5.释放资源(先关写的,后关闭读的;如果写完了,肯定读取完毕了)
        fos.close();
        fis.close();
        long e = System.currentTimeMillis();
        System.out.println("复制文件共耗时:"+(e-s)+"毫秒");
    }
}

file

file

file

file

//一次一字节太慢了,使用数组缓冲读取多个字节,写入多个字节
package InputOutput;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
public class Demo04 {
    public static void main(String[] args) throws IOException {
        long s = System.currentTimeMillis();
        //1.创建一个字节输入流对象,构造方法中绑定要读取的数据源
        FileInputStream fis =new FileInputStream("E:\\美图\\新建文件夹\\2.jpg");
        //2.创建一个字节输出流对象,构造方法中绑定要写入的目的地
        FileOutputStream fos =new FileOutputStream("E:\\美图\\2.jpg");
        byte[] bytes = new byte[1024];
        //3.使用字节输入流对象中的方法read读取文件
        int len = 0;//每次读取的有效字节个数
        while((len = fis.read(bytes))!=-1){
            //4.使用字节输出流中的方法write,把读取到的字节写入到目的地的文件中
            fos.write(bytes,0,len);
        }
        //5.释放资源(先关写的,后关闭读的;如果写完了,肯定读取完毕了)
        fos.close();
        fis.close();
        long e = System.currentTimeMillis();
        System.out.println("复制文件共耗时:"+(e-s)+"毫秒");
    }
}

file
file