Java基础学习之字节流(22)

1.1 IO流概述和分类

所有与IO操作相关的内容最后都要释放资源
IO流概述:

  • IO:输入/输出
  • 流:是一种抽象概念,是对数据传输的总称。流的本质就是数据传输
  • IO流就是用来处理设备间数据传输问题的
    常见应用:文件辅助、文件上传、文件下载

IO流分类

  • 按照数据流向
    输入流:读数据
    输出流:写数据
  • 按照数据类型来分
    字节流
    字节输入流;字节输出流
    字符流
    字符输入流;字符输出流

如果数据通过Window自带的记事本软件打开,我们还可以读懂里面的内容,就使用字符流,否则使用字节流;分不清就使用字节流

1.2 字节流写数据

字节流抽象基类

  • InputStream:这个抽象类是表示字节输入流的所有类的超类
  • OutputStream:这个抽象类是表示字节输出流的所有类的超类
  • 子类名特点:子类名称都是以其父类名作为子类名的后缀

FileOutputStream:文件输出流,用于将数据写入File
public FileOutputStream(String name):创建文件输出流以指定的名称写入文件

package java22;

import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

//public FileOutputStream(String name):创建文件输出流以指定的名称写入文件
public class demo1 {
    public static void main(String[] args) throws IOException {
        //创建字节输出流对象
        FileOutputStream f = new FileOutputStream("D:\\lyy\\f.txt");
        /*
        * 做了三件事情:
        *   调用系统功能创建文件
        *   创建字节输出流对象
        *   让字节输出流对象指向创建好的文件
        */
        //void write(int b):将指定的字节写入此文件输出流。
        f.write(99);
        f.write(57);//Ascall码57对应的数字9
        f.write(55);

        //最后都要释放资源
        //void close():关闭此文件输出流并释放与此流相关联的任何系统资源。
        f.close();
    }
}

使用字节输出流写数据的步骤:

  • 创建字节输出流对象(调用系统功能创建了文件、创建字节输出流对象、让字节输出流对象指向文件)
  • 调用字节输出流写对象的方法
  • 释放资源(关闭此文件输出流并释放与此流相关联的任何系统资源)

1.2.1 字节流写数据的三种方式

  • public void write(byte[] b) : 将 b.length字节从指定的字节数组写入此文件输出流。
  • public void write(int b) :将指定的字节写入此文件输出流。
  • public void write(byte[] b, int off, int len) : 从位于偏移量 off的指定字节数组写入 len字节到该文件输出流。
package java22;


import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
//构造方法
//FileOutputStream(File file):创建文件输出流以写入由指定的 File对象表示的文件。
//FileOutputStream(String name):创建文件输出流以指定的名称写入文件。

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


public class demo2 {
    public static void main(String[] args) throws IOException {
        //创建字节输出流
        FileOutputStream fs = new FileOutputStream("D:\\lyy\\fs.txt");
        // new File(name),相对于下面语句
//        FileOutputStream fs = new FileOutputStream(new File("D:\\lyy\\fs.txt"));

        //FileOutputStream(File file):创建文件输出流以写入由指定的 File对象表示的文件。
//        File file = new File("D:\\lyy\\fs.txt");
//        FileOutputStream fs2 = new FileOutputStream(file);
        //可以缩写为,fs和fs2的创建是等价的
//        FileOutputStream fs2 = new FileOutputStream(new File("D:\\lyy\\fs.txt"));

        //写数据的三种方式
        //public void write(int b)  :将指定的字节写入此文件输出流。
        fs.write(97);
        fs.write(98);
        fs.write(99);
        fs.write(100);

        //public void write(byte[] b) : 将 b.length字节从指定的字节数组写入此文件输出流
        byte[] by = {97,98,99,100};
        fs.write(by);
        //byte[] getBytes()  返回字符串对应的字节数组
        byte[] bys = "abcde".getBytes();
        fs.write(by);

        //public void write(byte[] b, int off, int len) : 从位于偏移量 off的指定字节数组写入 len字节到该文件输出流。
        fs.write(by,0,by.length);
        fs.write(by,1,3);//写入指定起始和末尾的字节
        
        //释放资源
        fs.close();
    }

}


1.2.2 字节流写数据的两个小问题

字节流写数据如何实现换行:
写完数据后,加换行符
window:\r\n
linux:\n
mac:\r

字节流数据如何实现追加写入:
public FileOutputStream(String name, boolean append)
创建文件输出流以指定的名称写入文件。如果第二个参数是true ,则字节将写入文件的末尾而不是开头。

package java22;

import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
//字节流写数据实现换行
//    window:\r\n
//    linux:\r
//    mac:\r

//如何实现追加写入
//public FileOutputStream(String name, boolean append)
//        创建文件输出流以指定的名称写入文件。
//        如果第二个参数是true ,则字节将写入文件的末尾而不是开头。

public class demo3 {
    public static void main(String[] args) throws IOException {
        //创建字节输出流
//        FileOutputStream fs = new FileOutputStream("D:\\lyy\\ja.txt");
        FileOutputStream fs = new FileOutputStream("D:\\lyy\\ja.txt",true);
        //写数据
        for(int i=0;i<10;i++){
            fs.write("hello".getBytes());
            //换行
            fs.write("\n".getBytes());
        }

        //释放资源
        fs.close();
    }
}

1.2.3 字节流写数据加异常处理

finally:在异常处理时提供finally块来执行所有清除操作。比如说IO流中的资源释放
特点:被finally控制的语句一定会执行,除非JVM退出

try{
可能出现异常的代码;
}catch(异常类名 变量名){
异常的处理代码;
}finally{
执行所有清除操作;
}
package java22;

import java.io.FileOutputStream;
import java.io.IOException;

public class demo4 {
    public static void main(String[] args) {
        //初始化变量
        FileOutputStream fs = null;
        try {
            //创建字节流输出
            fs = new FileOutputStream("D:\\lyy\\ja2.txt");
            fs.write("hello".getBytes());
        }catch (IOException e){
            e.printStackTrace();
        }finally {
            //判定fs是否为空,非空才进行释放操作,增强程序的健壮性
            if(fs != null) {
                try {
                    fs.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

1.3 字节流读数据(一次读一个字节数据)

FileInputStream:从文件系统中的文件获取输入字节。
构造方法:

  • public FileInputStream(String name):通过打开与实际文件的连接创建一个FileInputStream文件,该文件由文件系统中的路径名name命名。

使用字节输入流读数据的步骤:
创建字节输入流对象
调用字节输入流对象的读取方法
释放资源

package java22;

import java.io.FileInputStream;
import java.io.IOException;

public class demo5 {
    public static void main(String[] args) throws IOException {
        //创建字节输入流对象
        FileInputStream fs = new FileInputStream("D:\\lyy\\ja.txt");

        //调用字节输入流方法
        //public int read():从该输入流读取一个字节的数据。 如果没有输入可用,此方法将阻止。
        //数据的下一个字节,如果达到文件的末尾,则返回值 -1 。
        //第一次读取数据
//        int by = fs.read();
//        System.out.println(by);
//        System.out.println((char)by);
//        //第二次读取数据
//        by = fs.read();
//        System.out.println(by);
//        System.out.println((char)by);
        //采用循环读取
/*        int by = fs.read();
        while(by != -1){
            System.out.print((char)by);
            by = fs.read();
        }*/
        //优化程序
        int by;
        while ((by=fs.read()) != -1){
            System.out.print((char)by);
        }


        //释放资源
        fs.close();
    }
}

案例:复制文本文件

package java22;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;

public class demo6 {
    public static void main(String[] args) throws IOException {
        //根据数据源创建字节输入流对象
        FileInputStream fs = new FileInputStream("E:\\窗里窗外.txt");

        //根据目的地创建字节输出流对象
        FileOutputStream fos = new FileOutputStream("D:\\lyy\\窗里窗外.txt");

        //读写数据,复制文本文件(一次读取一个字节,一次写入一个字节)
        int by;
        while ((by=fs.read()) != -1){
            fos.write(by);
        }

        //释放资源
        fs.close();
        fos.close();
    }
}

1.3.1 字节流读数据(一次读一个字节组的数据)

package java22;

import java.io.FileInputStream;
import java.io.IOException;

public class demo7 {
    public static void main(String[] args) throws IOException {
        FileInputStream fis = new FileInputStream("D:\\lyy\\窗里窗外.txt");

//        //第一次读取
//        byte[] by = new byte[5];
//        int len = fis.read(by);
//        System.out.println(len);
        System.out.println(new String(by));
//        System.out.println(new String(by,0,len));
//        //第二次读取
//        len = fis.read(by);
//        System.out.println(len);
        System.out.println(new String(by));
//        System.out.println(new String(by,0,len));
//        //第三次读取
//        len = fis.read(by);
//        System.out.println(len);
//        //String(byte[] bytes, int offset, int length)
//        System.out.println(new String(by,0,len));
        /*
        daiji\r\n
        fdafe\r\n
        第一次读取:daiji
        第二次读取:\r\nfda
        所以输出结果为:
        daiji

        fda
        * */

        //循环改进
        byte[] by = new byte[1024];//1024及其整数倍
        int len;
        while ((len=fis.read(by))!=-1){
            System.out.println(new String(by,0,len));
        }


        fis.close();
    }
}

案例:复制图片

package java22;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;

//复制图片
public class demo8 {
    public static void main(String[] args) throws IOException {
        //创建字节输入流对象
        FileInputStream fis = new FileInputStream("C:\\Users\\lyy\\Desktop\\个人资料\\证件照\\证件照.jpg");
        //创建字节输出流对象
        FileOutputStream fos = new FileOutputStream("C:\\Users\\lyy\\Desktop\\个人资料\\证件照.jpg");

        byte[] by = new byte[1024];
        int len;
        while ((len=fis.read(by)) != -1){
            fos.write(by,0,len);
        }

        fos.close();
        fis.close();
    }
}

1.4 字节缓冲流

字节缓冲流:

  • BufferedOutputStream该类实现缓冲输出流。 通过设置这样的输出流,应用程序可以向底层输出流写入字节,而不必为写入的每个字节导致底层系统的调用。

  • 当创建BufferedInputStream时,将创建一个内部缓冲区数组。 当从流中读取或跳过字节时,内部缓冲区将根据需要从所包含的输入流中重新填充,一次有多个字节。

构造方法:

  • BufferedOutputStream(OutputStream out) :创建一个新的缓冲输出流,以将数据写入指定的底层输出流。
  • BufferedInputStream(InputStream in) :创建一个 BufferedInputStream并保存其参数,输入流 in ,供以后使用。
    为什么构造方法需要的是字节流,而不是具体的文件或者路径?
    字节缓冲流仅仅提供缓冲区,而真正的读写数据还得依靠基本的字节流对象进行操作。
package java22;

import java.io.*;

public class demo9 {
    public static void main(String[] args) throws IOException {
        //字节缓冲输出流
//        FileOutputStream fos = new FileOutputStream("D:\\lyy\\窗里窗外.txt");
//        BufferedOutputStream bos = new BufferedOutputStream(fos);
        //两个步骤合成一个步骤
        BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("D:\\lyy\\窗里窗外.txt"));
        //写数据
        bos.write("hello\r\n".getBytes());
        bos.write("world\r\n".getBytes());

        //释放资源
        bos.close();

        //字节缓冲输入流
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream("D:\\lyy\\窗里窗外.txt"));
        //读数据,一次读取一个字节
        int by;
        while((by=bis.read())!=-1){
            System.out.print((char) by);
        }
        //一次读取一个字节数组的数据
        byte[] bys = new byte[1024];
        int len;
        while ((len=bis.read(bys))!=-1){
            System.out.println(new String(bys,0,len));
        }
        //释放资源
        bis.close();

    }
}

四种读写文件方式的时间测试

  • 基本字节流一次读写一个字节
  • 基本字节流一次读写一个字节数组
  • 字符缓冲流一次读写一个字节
  • 字节缓冲流一次读写一个字节数组
package java22;

import java.io.*;

public class demo10 {
    public static void main(String[] args) throws IOException {
        //记录开始时间
        long startTime = System.currentTimeMillis();

        //复制图片
//        mathod1();
//        method2();
//        method3();
        method4();

        //记录结束时间
        long endTime = System.currentTimeMillis();
        System.out.println("共耗时:"+(endTime-startTime)+"ms");
    }

    //基本字节流一次读写一个字节,耗时16371ms
    public static void method1() throws IOException{
        //创建字节输入对象
        FileInputStream fis = new FileInputStream("C:\\Users\\lyy\\Desktop\\个人资料\\证件照\\证件照.jpg");

        //创建输出流对象
        FileOutputStream fos = new FileOutputStream("C:\\Users\\lyy\\Desktop\\个人资料\\证件照.jpg");

        //一次读写一个字节
        int by;
        while ((by=fis.read())!=-1){
            fos.write(by);
        }
        fis.close();
        fos.close();
    }

    //基本字节流一次读写一个字节数组,耗时24ms
    public static void method2() throws IOException{
        FileInputStream fis = new FileInputStream("C:\\Users\\lyy\\Desktop\\个人资料\\证件照\\证件照.jpg");

        //创建输出流对象
        FileOutputStream fos = new FileOutputStream("C:\\Users\\lyy\\Desktop\\个人资料\\证件照.jpg");

        //一次读写一个字节数组
        byte[] bys = new byte[1024];
        int len;
        while ((len=fis.read(bys))!=-1){
            fos.write(bys,0,len);
        }
        fis.close();
        fos.close();
    }

    //字节缓冲流一次读写一个字节,共耗时96ms
    public static void method3() throws IOException{
        //创建字节输入对象
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream("C:\\Users\\lyy\\Desktop\\个人资料\\证件照\\证件照.jpg"));

        //创建输出流对象
        BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("C:\\Users\\lyy\\Desktop\\个人资料\\证件照.jpg"));

        //一次读写一个字节
        int by;
        while ((by=bis.read())!=-1){
            bos.write(by);
        }
        bis.close();
        bos.close();
    }

    //字节缓冲流一次读写一个字节数组,共耗时4ms
    public static void method4() throws IOException{
        //创建字节输入对象
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream("C:\\Users\\lyy\\Desktop\\个人资料\\证件照\\证件照.jpg"));

        //创建输出流对象
        BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("C:\\Users\\lyy\\Desktop\\个人资料\\证件照.jpg"));

        //一次读写一个字节数组
        byte[] bys = new byte[1024];
        int len;
        while ((len=bis.read(bys))!=-1){
            bos.write(len);
        }
        bis.close();
        bos.close();
    }
}

总结:尽量选择字节缓冲流一次读写一个字节数组

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值