今日内容介绍

  • 字节流
  • 字符流

    1 字节流

在前面的学习过程中,我们一直都是在操作文件或者文件夹,并没有给文件中写任何数据。现在我们就要开始给文件中写数据,或者读取文件中的数据。

1.1 字节输出流OutputStream

OutputStream此抽象类,是表示输出字节流的所有类的超类。操作的数据都是字节,定义了输出字节流的基本共性功能方法。

输出流中定义都是写write方法,如下图:

1.1.1 FileOutputStream类

OutputStream有很多子类,其中子类FileOutputStream可用来写入数据到文件。

FileOutputStream类,即文件输出流,是用于将数据写入 File的输出流。

  • 构造方法

1.1.2FileOutputStream类写入数据到文件中

  • 将数据写到文件中,代码演示:
public class FileOutputStreamDemo {

    public static void main(String[] args) throws IOException {

        //需求:将数据写入到文件中。

        //创建存储数据的文件。

        File file = new File("c:\\file.txt");

        //创建一个用于操作文件的字节输出流对象。一创建就必须明确数据存储目的地。

        //输出流目的是文件,会自动创建。如果文件存在,则覆盖。

        FileOutputStream fos = new FileOutputStream(file);

        //调用父类中的write方法。

        byte[] data = "abcde".getBytes();

        fos.write(data);

        //关闭流资源。

        fos.close();

    }

}

1.1.3 给文件中续写和换行

我们直接new FileOutputStream(file)这样创建对象,写入数据,会覆盖原有的文件,那么我们想在原有的文件中续写内容怎么办呢?

继续查阅FileOutputStream的API。发现在FileOutputStream的构造函数中,可以接受一个boolean类型的值,如果值true,就会在文件末位继续添加。

  • 构造方法

  • 给文件中续写数据和换行,代码演示:
public class FileOutputStreamDemo2 {

    public static void main(String[] args) throws Exception {

        File file = new File("c:\\file.txt");

        fos.write(str.getBytes());

        fos.close();

    }

}

IO异常的处理

在前面编写代码中都发生了IO的异常。我们在实际开发中,对异常时如何处理的,我们来演示一下。

public class FileOutputStreamDemo3 {

    public static void main(String[] args) {

        File file = new File("c:\\file.txt");

        //定义FileOutputStream的引用

        FileOutputStream fos = null;

        try {

            //创建FileOutputStream对象

            fos = new FileOutputStream(file);

            //写出数据

            fos.write("abcde".getBytes());

        } catch (IOException e) {

            System.out.println(e.toString() + "----");

        } finally {

            //一定要判断fos是否为null,只有不为null时,才可以关闭资源

            if (fos != null) {

                try {

                    fos.close();

                } catch (IOException e) {

                    throw new RuntimeException("");

                }

            }

        }

    }

}

1.2 字节输入流InputStream

通过前面的学习,我们可以把内存中的数据写出到文件中,那如何想把内存中的数据读到内存中,我们通过InputStream可以实现。InputStream此抽象类,是表示字节输入流的所有类的超类。,定义了字节输入流的基本共性功能方法。

  • int read():读取一个字节并返回,没有字节返回-1.
  • int read(byte[]): 读取一定量的字节数,并存储到字节数组中,返回读取到的字节数。

1.2.1 FileInputStream类

InputStream有很多子类,其中子类FileInputStream可用来读取文件内容。

FileInputStream 从文件系统中的某个文件中获得输入字节。

  • 构造方法

1.2.2 FileInputStream类读取数据read方法

在读取文件中的数据时,调用read方法,实现从文件中读取数据

  • 从文件中读取数据,代码演示:
public class FileInputStreamDemo {

    public static void main(String[] args) throws IOException {

        File file = new File("c:\\file.txt");

        //创建一个字节输入流对象,必须明确数据源,其实就是创建字节读取流和数据源相关联。

        FileInputStream fis = new FileInputStream(file);

        //读取数据。使用 read();一次读一个字节。

        int ch = 0;

        while((ch=fis.read())!=-1){

            System.out.pr        }intln("ch="+(char)ch);

        // 关闭资源。

        fis.close();

    }

}

1.2.3 读取数据read(byte[])方法

在读取文件中的数据时,调用read方法,每次只能读取一个,太麻烦了,于是我们可以定义数组作为临时的存储容器,这时可以调用重载的read方法,一次可以读取多个字符。

public class FileInputStreamDemo2 {

    public static void main(String[] args) throws IOException {

        /*

         * 演示第二个读取方法, read(byte[]);

         */

        File file = new File("c:\\file.txt");

        // 创建一个字节输入流对象,必须明确数据源,其实就是创建字节读取流和数据源相关联。

        FileInputStream fis = new FileInputStream(file);        

        //创建一个字节数组。

        byte[] buf = new byte[1024];//长度可以定义成1024的整数倍。        

        int len = 0;

        while((len=fis.read(buf))!=-1){

            System.out.println(new String(buf,0,len));

        }

        fis.close();

    }

}

1.3 字节流练习

既然会了文件的读和写操作了,那么我们就要在这个基础上进行更为复杂的操作。使用读写操作完成文件的复制。

1.3.1 复制文件

原理;读取一个已有的数据,并将这些读到的数据写入到另一个文件中。

public class CopyFileTest {

    public static void main(String[] args) throws IOException {

        //1,明确源和目的。

        File srcFile = new File("c:\\YesDir\test.JPG");

        File destFile = new File("copyTest.JPG");

        //2,明确字节流输入流和源相关联,输出流和目的关联。

        FileInputStream fis = new FileInputStream(srcFile);

        FileOutputStream fos = new FileOutputStream(destFile);

        //3, 使用输入流的读取方法读取字节,并将字节写入到目的中。

        int ch = 0;

        while((ch=fis.read())!=-1){

            fos.write(ch);

        }

        //4,关闭资源。

        fos.close();

        fis.close();

    }

}

上述代码输入流和输出流之间是通过ch这个变量进行数据交换的。

上述复制文件有个问题,每次都从源文件读取一个,然后在写到指定文件,接着再读取一个字符,然后再写一个,一直这样下去。效率极低。

1.3.2 缓冲数组方式复制文件

上述代码复制文件效率太低了,并且频繁的从文件读数据,和写数据,能不能一次多把文件中多个数据都读进内容中,然后在一次写出去,这样的速度一定会比前面代码速度快。

public class CopyFileByBufferTest {

    public static void main(String[] args) throws IOException {

        File srcFile = new File("c:\\YesDir\test.JPG");

        File destFile = new File("copyTest.JPG");

        // 明确字节流输入流和源相关联,输出流和目的关联。

        FileInputStream fis = new FileInputStream(srcFile);

        FileOutputStream fos = new FileOutputStream(destFile);

        //定义一个缓冲区。

        byte[] buf = new byte[1024];

        int len = 0;

        while ((len = fis.read(buf)) != -1) {

            fos.write(buf, 0, len);// 将数组中的指定长度的数据写入到输出流中。

        }

        // 关闭资源。

        fos.close();

        fis.close();

    }

}

2 字符流

经过前面的学习,我们基本掌握的文件的读写操作,在操作过程中字节流可以操作所有数据,可是当我们操作的文件中有中文字符,并且需要对中文字符做出处理时怎么办呢?

2.1 字节流读取字符的问题

通过以下程序读取带有中文件的文件。

public class CharStreamDemo {

    public static void main(String[] args) throws IOException {

        //给文件中写中文

        writeCNText();

        //读取文件中的中文

        readCNText();

    }    

    //读取中文

    public static void readCNText() throws IOException {

        FileInputStream fis = new FileInputStream("c:\\cn.txt");

        int ch = 0;

        while((ch = fis.read())!=-1){

            System.out.println(ch);

        }

    }

    //写中文

    public static void writeCNText() throws IOException {

        FileOutputStream fos = new FileOutputStream("c:\\cn.txt");

        fos.write("a传智播客欢迎你".getBytes());

        fos.close();

    }

}

上面程序在读取含有中文的文件时,我们并没有看到具体的中文,而是看到一些数字,这是什么原因呢?既然看不到中文,那么我们如何对其中的中文做处理呢?要解决这个问题,我们必须研究下字符的编码过程。

2.2 字符编码表

我们知道计算机底层数据存储的都是二进制数据,而我们生活中的各种各样的数据,如何才能和计算机中存储的二进制数据对应起来呢?

这时老美他们就把每一个字符和一个整数对应起来,就形成了一张编码表,老美他们的编码表就是ASCII表。其中就是各种英文字符对应的编码。

编码表:其实就是生活中字符和计算机二进制的对应关系表。

1、ascii: 一个字节中的7位就可以表示。对应的字节都是正数。0-xxxxxxx

2、iso-8859-1:拉丁码表 latin,用了一个字节用的8位。1-xxxxxxx 负数。

3、GB2312:简体中文码表。包含6000-7000中文和符号。用两个字节表示。两个字节第一个字节是负数,第二个字节可能是正数

GBK:目前最常用的中文码表,2万的中文和符号。用两个字节表示,其中的一部分文字,第一个字节开头是1,第二字节开头是0

GB18030:最新的中文码表,目前还没有正式使用。

  1. unicode:国际标准码表:无论是什么文字,都用两个字节存储。
  • Java中的char类型用的就是这个码表。char c = 'a';占两个字节。
  • Java中的字符串是按照系统默认码表来解析的。简体中文版 字符串默认的码表是GBK。

5、UTF-8:基于unicode,一个字节就可以存储数据,不要用两个字节存储,而且这个码表更加的标准化,在每一个字节头加入了编码信息(后期到api中查找)。

能识别中文的码表:GBK、UTF-8;正因为识别中文码表不唯一,涉及到了编码解码问题。

对于我们开发而言;常见的编码 GBK UTF-8 ISO-8859-1

文字--->(数字) :编码。 "abc".getBytes() byte[]

(数字)--->文字 : 解码。 byte[] b={97,98,99} new String(b)

2.3 字符输入流Reader

上述程序中我们读取拥有中文的文件时,使用的字节流在读取,那么我们读取到的都是一个一个字节。只要把这些字节去查阅对应的编码表,就能够得到与之对应的字符。API中是否给我们已经提供了读取相应字符的功能流对象,Reader,读取字符流的抽象超类。

  • read():读取单个字符并返回
  • read(char[]):将数据读取到数组中,并返回读取的个数。

2.3.1 FileReader类

查阅FileInputStream的API,发现FileInputStream 用于读取诸如图像数据之类的原始字节流。要读取字符流,请考虑使用 FileReader。

打开FileReader的API介绍。用来读取字符文件的便捷类。此类的构造方法假定默认字符编码和默认字节缓冲区大小都是适当的

  • 构造方法

2.3.2 FileReader读取包含中文的文件

  • 使用FileReader读取包含中文的文件
public class CharStreamDemo {

    public static void main(String[] args) throws IOException {

        //给文件中写中文

        writeCNText();

        //读取文件中的中文

        readCNText();

    }    

    //读取中文

    public static void readCNText() throws IOException {

        FileReader fr = new FileReader("D:\\test\\cn.txt");

        int ch = 0;

        while((ch = fr.read())!=-1){

            //输出的字符对应的编码值

            System.out.println(ch);

            //输出字符本身

            System.out.println((char)ch);

        }

    }

    //写中文

    public static void writeCNText() throws IOException {

        FileOutputStream fos = new FileOutputStream("D:\\test\\cn.txt");

        fos.write("a传智播客欢迎你".getBytes());

        fos.close();

    }

}

2.4 字符输出流Writer

既然有专门用于读取字符的流对象,那么肯定也有写的字符流对象,查阅API,发现有一个Writer类,Writer是写入字符流的抽象类。其中描述了相应的写的动作。

2.4.1 FileWriter类

查阅FileOutputStream的API,发现FileOutputStream 用于写入诸如图像数据之类的原始字节的流。要写入字符流,请考虑使用 FileWriter。

打开FileWriter的API介绍。用来写入字符文件的便捷类。此类的构造方法假定默认字符编码和默认字节缓冲区大小都是可接受的。

  • 构造方法

2.4.2 FileWriter写入中文到文件中

  • 写入字符到文件中,先进行流的刷新,再进行流的关闭。
public class FileWriterDemo {

    public static void main(String[] args) throws IOException {

        //演示FileWriter 用于操作文件的便捷类。

        FileWriter fw = new FileWriter("d:\\text\\fw.txt");

        fw.write("你好谢谢再见");//这些文字都要先编码。都写入到了流的缓冲区中。

        fw.flush();

        fw.close();

    }

}

2.5 flush()和close()的区别?

flush():将流中的缓冲区缓冲的数据刷新到目的地中,刷新后,流还可以继续使用。

close():关闭资源,但在关闭前会将缓冲区中的数据先刷新到目的地,否则丢失数据,然后在关闭流。流不可以使用。如果写入数据多,一定要一边写一边刷新,最后一次可以不刷新,由close完成刷新并关闭。

2.6 字符流练习

2.6.1 复制文本文件

练习:复制文本文件。

思路:

1,既然是文本涉及编码表。需要用字符流。

2,操作的是文件。涉及硬盘。

3,有指定码表吗?没有,默认就行。

操作的是文件,使用的 默认码表。使用哪个字符流对象。直接使用字符流操作文件的便捷类。FileReader FileWriter

public class CopyTextFileTest {

    public static void main(String[] args) throws IOException {

        copyTextFile();

    }

    public static void copyTextFile() throws IOException {

        //1,明确源和目的。

        FileReader fr = new FileReader("c:\\cn.txt");

        FileWriter fw = new FileWriter("c:\\copy.txt");

        //2,为了提高效率。自定义缓冲区数组。字符数组。

        char[] buf = new char[1024];

        int len = 0;

        while((len=fr.read(buf))!=-1){

            fw.write(buf,0,len);

        }

        /*2,循环读写操作。效率低。

        int ch = 0;

        while((ch=fr.read())!=-1){

            fw.write(ch);

        }

        */

        //3,关闭资源。

        fw.close();

        fr.close();

    }

}

3 总结

3.1 知识点总结

  • IO流的分类

    |- 字节流

    |- 字节输入流 InputStream 抽象类

    |- FileInputStream 操作文件的字节输入流

    |- 字节输出流 OuputStream抽象类

    |- FileOutputStream 操作文件的字节输出流

    |- 字符流

    |- 字符输入流 Reader抽象类

    |- InputStreamReader 输入操作的转换流

    |- FileReader 用来操作文件的字符输入流(简便的流)

    |- 字符输出流 Writer抽象类

    |- OutputStreamWriter 输出操作的转换流

    |- FileWriter 用来操作文件的字符输出流(简便的流)

Java学习IO流第一天的更多相关文章

  1. java学习——IO流

    字符流的由来:其实就是:字节流读取文字字节数据后,不直接操作而是先查指定的编码表.获取对应的文字.在对这个文字进行操作.简单说:字节流+编码表 ---------------------------- ...

  2. java的Io流学习

    Java中io流的学习(一)File:https://blog.csdn.net/qq_41061437/article/details/81672859 Java中io流的学习(二)FileInpu ...

  3. Java基础——IO流

    今天刚刚看完java的io流操作,把主要的脉络看了一遍,不能保证以后使用时都能得心应手,但是最起码用到时知道有这么一个功能可以实现,下面对学习进行一下简单的总结: IO流主要用于硬板.内存.键盘等处理 ...

  4. 学习IO流

    学习IO流,不得不提到的就是JavaIO流.流就是字节序列的抽象概念,能被连续读取数据的数据源和能被连续写入数据的接受端就是流,流机制是Java及C++中的一个重要机制,通过流我们可以自由得控制文件, ...

  5. JAVA中IO流总结

    本文是在学习中的总结,欢迎转载但请注明出处:http://blog.csdn.net/pistolove/article/details/42119261 我想你对JAVA的IO流有所了解,平时使用的 ...

  6. Java基础IO流(二)字节流小案例

    JAVA基础IO流(一)https://www.cnblogs.com/deepSleeping/p/9693601.html ①读取指定文件内容,按照16进制输出到控制台 其中,Integer.to ...

  7. Java基础-IO流对象之字节缓冲流(BufferedOutputStream与BufferedInputStream)

    Java基础-IO流对象之字节缓冲流(BufferedOutputStream与BufferedInputStream) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 在我们学习字 ...

  8. Java基础-IO流对象之字符类(FileWrite与FileReader)

    Java基础-IO流对象之字符类(FileWrite与FileReader) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.常见编码简介 1>ASCII 我们知道计算机是 ...

  9. java的IO流,字节流和字符流

    java操作文件都是通过流来处理的,(其实其他很多语言也是这样) 第一:java的IO流,分为:输入流 和 输出流(这真是废话,这是从流向的角度来说的) 第二:java的所有IO流,只分为:字节流 和 ...

随机推荐

  1. Django Admin后台管理功能使用+二次开发

    一  使用环境 开发系统: windows IDE: pycharm 数据库: msyql,navicat 编程语言: python3.7  (Windows x86-64 executable in ...

  2. 「TcaplusDB知识库」概念(表、键、记录、索引)

       TcaplusDB作为一款NoSQL数据库,语法与传统的SQL关系库有所差异.本文将详细介绍TcaplusDB表.记录.索引这三个数据库中常用术语在TcaplusDB中的概念与意义. 术语\概念 ...

  3. SpringCloud Sleuth

    1.定义 Sleuth(分布式请求链路跟踪):提供了一套完整的服务跟踪解决方案,也兼容zipkin. 参考网址:https://github.com/spring-cloud/spring-cloud ...

  4. vue-axios插件、django-cors插件、及vue如何使用第三方前端样式库:element/jQuery/bootstrap

    目录 一.vue的ajax插件:axios 1.安装axios 2.axios参数 二.CORS跨域问题(同源策略) 1.Django解决CORS跨域问题方法 三.前端请求携带参数及Django后台如 ...

  5. redis基础:redis下载安装与配置,redis数据类型使用,redis常用指令,jedis使用,RDB和AOF持久化

    知识点梳理 课堂讲义 课程计划 1. REDIS 入 门 (了解) (操作)   2. 数据类型 (重点) (操作) (理解) 3. 常用指令   (操作)   4. Jedis (重点) (操作) ...

  6. 《从零开始TypeScript》系列 - 基础数据类型

    TypeScript 是 JavaScript 的超集,这里我们只讨论两者中的不同的部分,或者需要注意的部分 数组 Array:在TypeScript中,有两种方式来定义一个数组: 在元素类型后面接上 ...

  7. Python基础(2)——循环和分支[xiaoshun]

    一.瞎扯 世界上一切的系统都可以被'分支'表示.循环也是分支,只不过又重复之前的'分支'选择罢了.程序如人生,每一次的'分支',每一次的选择,都会有不同的结果: 有的选择止步不前,无限循环: 有的选择 ...

  8. github个人主页 阿里云域名的绑定

    域名解析 我在阿里云上买了一个新域名:gaolu.name,我已经在GitHub Pages上建立了自己的博客:http://gaolu1215.github.io.现在我希望将gaolu.name映 ...

  9. 2.掌握numpy数组

    一.改变数组形态 reshape()--通过改变数组的维度改变数组形态 import numpy as np Array=np.arange(1,17,1) Array Array_1=np.aran ...

  10. 这个Bug的排查之路,真的太有趣了。

    这是why哥的第 92 篇原创文章 在<深入理解Java虚拟机>一书中有这样一段代码: public class VolatileTest {    public static volat ...