IO流

纵然是瞬间 提交于 2020-08-14 12:33:44

IO流

File

File类概述

File:它是文件和目录路径名的抽象表示

  • 文件和目录是可以通过File封装成对象的
  • 对于File而言,其封装性并不是一个真正存在的文件,仅仅是一个路径名而已。它可以是存在的,也可以是不存在的,将来要是用过具体的操作把这个路径的内容转换为具体存在的

构造方法

  1. File(String pathname) 通过将给定的路径名字符串转换为抽象路径名来创建新的File实例
  2. File(String parent,String child) 从路径名字字符串和子路径名字字符串创建新的File实例
  3. File(File parent,String child)从父抽象路径名和子路径名字字符串创建新的File实例
import java.io.File;
public class Demo {
    public static void main(String[] args) {
        //File(String pathname) 通过将给定的路径名字符串转换为抽象路径名来创建新的File实例
        File f1 = new File("D:\\JAVA\\java.txt");
        System.out.println(f1);
        //File(String parent,String child) 从路径名字字符串和子路径名字字符串创建新的File实例
        File f2 = new File("D:\\JAVA","java.txt");
        System.out.println(f2);
        //File(File parent,String child)从父抽象路径名和子路径名字字符串创建新的File实例
        File f3 = new File("D:\\JAVA");
        File f4 = new File(f3,"java.txt");
        System.out.println(f4);
    }
}

File类创建功能

  1. public boolean createNewFile() 当具有该名称的文件不存在时,创建一个由该抽象路径命名的新空文件

    • 文件名存在,就不创建,返回false
    • 文件名不存在,就创建,返回ture
  2. public boolean mkdir() 创建由此路径名命名的目录

    • 目录名存在,就不创建,返回false
    • 目录名不存在,就创建,返回ture
  3. punlic boolean mkdirs() 创建由此抽象路径名命名的目录,包括任何必须但不存在的父目录

    • 目录名存在,就不创建,返回false
    • 目录名不存在,就创建,返回ture
import java.io.File;
import java.io.IOException;
public class Demo {
    public static void main(String[] args) throws IOException {
        //在D:\\JAVA目录下创建一个文件java.txt
        File f1 = new File("D:\\JAVA\\java.txt");
        System.out.println(f1.createNewFile());
        //在E:\\JAVA目录下创建一个目录JavaEE
        System.out.println("----------------");
        File f2 = new File("D:\\JAVA\\JAVAEE");
        System.out.println(f2.mkdir());
        //在E:\\JAVA目录下创建一个多级目录javaWEB\\HTML
        System.out.println("----------------");
        File f3 = new File("D:\\JAVA\\javaWEB\\HTML");
        System.out.println(f3.mkdirs());
    }
}


File判断和获取功能

  1. public boolean isDiectory() 测试此抽象路径名表示的File是否为目录
  2. public boolean isFiel() 测试此抽象路径名表示的Fiel是否为文件
  3. public boolean exists() 测试此抽象路径名表示的File是否存在
  4. public String getAbsolutePath() 返回此抽象路径名转换为路径名字字符串
  5. public String getPath() 将此抽象路径名转换为路径名字符串
  6. public String getName() 返回由此抽象路径名表示的文件或目录的名称
  7. public String[] list() 返回此抽象路径名表示的目录中的文件和目录的名称字符串数组
  8. public File[] listFiles() 返回此抽象路径名表示的目录中的文件和目录的File对象数组
import java.io.File;
public class Demo {
    public static void main(String[] args) {
//        1. public boolean isDiectory() 测试此抽象路径名表示的File是否为目录
//        1. public boolean isFiel() 测试此抽象路径名表示的Fiel是否为文件
//        3. public boolean exists() 测试此抽象路径名表示的File是否存在
        File f1 = new File("D:\\JAVA\\java.txt");
        System.out.println(f1.isDirectory());
        System.out.println(f1.isFile());
        System.out.println(f1.exists());
        System.out.println("---------------------------");
//        4. public String getAbsolutePath() 返回此抽象路径名转换为路径名字字符串
//        5. public String getPath() 将此抽象路径名转换为路径名字符串
//        6. public String getName() 返回由此抽象路径名表示的文件或目录的名称
        System.out.println(f1.getAbsolutePath());
        System.out.println(f1.getPath());
        System.out.println(f1.getName());
//        7. public String[] list() 返回此抽象路径名表示的目录中的文件和目录的名称字符串数组
//       8. public File[] listFiles() 返回此抽象路径名表示的目录中的文件和目录的File对象数组
        File f2 = new File("D:\\JAVA");
        String[] list = f2.list();
        System.out.println(list);
        for (String li : list) {
            System.out.println(li);
        }
        System.out.println("---------------------");
        File[] files = f2.listFiles();
        for (File filelist : files) {
//            System.out.println(filelist);
            if (filelist.isFile()) {
                System.out.println(filelist.getName());
            }
        }
    }
}


File类删除功能

  • public boolean delete() 删除由此抽象路径名表示的文件或目录

遍历目录

/*
给定一个路径(E:\\JAVA),请通过递归完成遍历该目录下的所有内容,并把所有文件的绝对路径输出到控制台

思路:
根据给定路径创建一个File对象
定义一个方法用于获取给定目录下的所有内容,参数为第一步创建的File对象
获取给定的File目录下所有的文件或者目录的File数组
遍历该File数组,得到每一个File对象
判断该File对象是否是目录
    是:递归调用
    不是:获取绝对路径输出在控制台
调用方法
 */
import java.io.File;
public class Demo {
    public static void main(String[] args) {
        File fiel = new File("D:\\JAVA");
        getAllFilePath(fiel);
    }
    public static void getAllFilePath(File file){
        File[] files = file.listFiles();
        if (files != null){
            for (File i : files){
                if (i.isDirectory()){
                    getAllFilePath(i);
                }else{
                    System.out.println(i.getAbsolutePath());
                }
            }
        }
    }
}

IO流概述和分类

IO流概述:

  • IO:输入/输出(input/Output)
  • 流 : 是一种抽象概念,是对数据传输的总称。也就是说数据在设备间的传输称为流,流的本质是数据传输
  • IO流就是用来处理设备间数据传输问题的
    • 常见的应用:文件复制;文件上传;文件下载

分类

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

使用场景:

  • 如果数据通过Windows自带的记事本,我们还可以读懂里面的内容,用字符流;否则用字节流。若不知道使用哪种类型,用字节流(万能流)。

字节流

字节流抽象基类

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

FileOutputStream:文件输出流用于将数据写入File

  • FileOutputStream(String name):创建文件输出流以指定的名称写入文件

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

  1. 创建字节输出流对象(调用系统功能创建了文件,创建字节输出流对象,让字节输出流对象指向文件)
  2. 调用字节输出流对象的写数据方法
  3. 释放资源(关闭此文件输出流并释放与此相关联的任何系统资源)
/*
FileOutputStream(String name):创建文件输出流以指定的名称写入文件
     做了三件事:
        调用系统功能创建了文件
        创建了字节输出流对象
        让字节输出流对象指向创建好的文件
void write(int b):将指定的字节写入此文件输出流
最后都要释放资源
void colse():关闭此文件输出流并释放与此相关联的任何系统资源
 */
 import java.io.FileOutputStream;
import java.io.IOException;
public class Demo {
    public static void main(String[] args) throws IOException {
        FileOutputStream fos = new FileOutputStream("fos.txt");
        fos.write(97);
        fos.close();
    }
}

字节流写数据的三种方式

  • void write(int b) 将指定的字节写入此文件输出流
  • void write(byte[] b)将b.length字节从指定的字节数组写入此文件输出流一次写一个字节数组数据
  • void write(byte[] b, int off, int len)将len字节从指定的字节数组开始,从偏移量off(索引位置)开始写入此文件输出流一次写一个字节数组的部分数据
import java.io.FileOutputStream;
import java.io.IOException;
public class Demo {
    public static void main(String[] args) throws IOException {
        FileOutputStream fos = new FileOutputStream("fos.txt");
//void write(int b) 将指定的字节写入此文件输出流
//        fos.write(97);
//        fos.write(98);
//        fos.write(99);
//        fos.write(100);
//        fos.write(101);

//void write(byte[] b)将b.length字节从指定的字节数组写入此文件输出流一次写一个字节数组数据
     byte[] bytes = "abcde".getBytes();
//        fos.write(bytes);

//void write(byte[] b, int off, int len)将len字节从指定的字节数组开始,从偏移量off(索引位置)开始写入此文
        fos.write(bytes,1,3);
    }
}
fos.colse();

写入数据的两个问题

  • 如何换行
    • Windows: \r\n
    • Linux: \n
    • mac: \r
  • 如何追加写入
    • public FileOutputStream​(String name,boolean append)
      创建文件输出流以指定的名称写入文件。
      如果第二个参数为true ,则字节将写入文件的末尾而不是开头。


import java.io.FileOutputStream;
import java.io.IOException;
public class Demo {
    public static void main(String[] args) throws IOException {
        FileOutputStream fos = new FileOutputStream("IO流\\fos.txt",true);
        //如何换行
        for (int i = 0; i < 10; i++){
            fos.write("hello".getBytes());
            //如何换行
            fos.write("\r\n".getBytes());
        }
        //释放资源
    fos.close();
    }
}


字节流写数据+异常处理

  • finally:在异常处理时提供finally块来执行所有请出操作。比如说IO流中的释放资源
    • 特点:被finally控制的语句一定会执行,除非JVM退出
  • 格式:
try{
    可能出现异常的代码
}catch(异常类名 变量名){
    异常的处理代码;
}finally{
    执行所有清除操作;
}

import java.io.FileOutputStream;
import java.io.IOException;
public class Demo {
    public static void main(String[] args) {
        FileOutputStream fos = null;
        try{
            fos = new FileOutputStream("IO流\\fos.txt");
            fos.write("hello".getBytes());
        }catch (IOException e){
            e.printStackTrace();
        }finally {
            if (fos != null){
                try{
                    fos.close();
                }catch (IOException e){
                    e.printStackTrace();
                }
            }
        }
    }
}

字节流读数据

FileInputStream:文件系统中的获取输入字节

  • int read(),从该输入流读取一个字节的数据
  • int read​(byte[] b) 从该输入流读取最多 b.length个字节的数据到一个字节数组。
  • int read​(byte[] b, int off, int len) 从该输入流读取最多 len个字节的数据到字节数组。
  • 使用字节输入流读数据的步骤:
    • 创建字节输入流对象
    • 调用字节输入流对象的读数据方法
    • 释放资源

字节缓冲流

  • BufferedOutputStream:该类实现缓冲输出流。通过设置这样的输出流,应用程序可以向底层输出流写入字节,而不必为写入的每个字节导致底层系统的调用
  • BufferedInputStream:创建BufferedInputStream将创建一个内部缓冲区数组。当从流中读取或者跳过字节时,内部缓冲区将根据需要从所包含的输入流中重新填充,一次很多字节。

构造方法

  • 字节缓冲输出流:BufferedOutputStream(OutputStream out)
  • 字节缓冲输入流:BufferedInputStream(InputStream in)

为什么构造方法需要的是字节流,而不是具体的文件或者路径呢?

  • 字节缓冲流仅仅提供缓冲区,而真正的读写数据还得依靠基本的字节流对象进行操作。
package 字节流.字节缓冲流;

import java.io.*;

/*
 * BufferedOutputStream
 * * BufferedInputStream
 * 构造方法:
 * 字节缓冲输出流:BufferedOutputStream(OutputStream out)
 * 字节缓冲输入流:BufferedInputStream(InputStream in)
 */
public class Demo {
    public static void main(String[] args) {
        //BufferedOutputStream
/*        BufferedOutputStream bos = null;
    try {
        bos = new BufferedOutputStream(new FileOutputStream("bos.txt"));
        bos.write("hello\r\n".getBytes());
        bos.write("world\r\n".getBytes());
    }catch (IOException e){
        e.printStackTrace();
    }finally {
        if (bos != null){
            try {
                bos.close();
            }catch (IOException e){
                e.printStackTrace();
            }
        }
    }*/

//BufferedInputStream
        //读一
/*        BufferedInputStream bis = null;
        try {
            bis = new BufferedInputStream(new FileInputStream("bos.txt"));
            int len;
           while((len = bis.read()) != -1){
           System.out.print((char)len);
           }
        }catch (IOException e){
            e.printStackTrace();
        }finally {
            if (bis != null){
                try {
                    bis.close();
                }catch (IOException e){
                    e.printStackTrace();
                }
            }
        }*/
//读二
        BufferedInputStream bis = null;
        try {
            bis = new BufferedInputStream(new FileInputStream("bos.txt"));
            byte [] b = new byte[1024];
            int len;
            while ((len = bis.read(b)) != -1){
                System.out.print(new String(b,0,len));
            }
        }catch (IOException e){
            e.printStackTrace();
        }finally {
            if (bis != null) {
                try {
                    bis.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}


复制视频效率对比:


import java.io.*;

public class Demo {
    public static void main(String[] args) {
        long start = System.currentTimeMillis();
        //缓冲字节流一次读取一个字节
//        method();                         // 50 ms
        //缓冲字节流一次读取一个字节数组
        method2();                        // 2 ms
        //基本字节流一次读取一个字节数组
//        method3();                        // 20 ms
        //基本字节流一次读取一个字节
//        method4();                         //  6000 ms
        long end = System.currentTimeMillis();
        System.out.println(end - start + "ms");
    }

    //缓冲字节流一次读取一个字节
    public static void method() {
        BufferedInputStream bis = null;
        BufferedOutputStream bos = null;
        try {
            bis = new BufferedInputStream(new FileInputStream("D:\\JAVA\\NBA.qlv"));
            bos = new BufferedOutputStream(new FileOutputStream("D:\\JAVA\\JAVAEE\\NBA.qlv"));
            int len;
            while ((len = bis.read()) != -1) {
                bos.write(len);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (bis == null || bos == null) {
                try {
                    bis.close();
                    bos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    //缓冲字节流一次读取一个字节数组
    public static void method2() {
        BufferedInputStream bis = null;
        BufferedOutputStream bos = null;
        try {
            bis = new BufferedInputStream(new FileInputStream("D:\\JAVA\\NBA.qlv"));
            bos = new BufferedOutputStream(new FileOutputStream("D:\\JAVA\\JAVAEE\\NBA.qlv"));
            byte[] b = new byte[1024];
            int len;
            while ((len = bis.read(b)) != -1) {
                bos.write(b,0,len);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (bis == null || bos == null) {
                try {
                    bis.close();
                    bos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    //基本字节流一次读取一个字节数组
    public static void method3() {
        FileOutputStream fos = null;
        FileInputStream fis = null;
        try {
            fis = new FileInputStream("D:\\JAVA\\NBA.qlv");
            fos = new FileOutputStream("D:\\JAVA\\JAVAEE\\NBA.qlv");
            byte[] b = new byte[1024];
            int len;
            while ((len = fis.read(b)) != -1) {
                fos.write(b,0,len);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (fos == null || fis == null) {
                try {
                    fos.close();
                    fis.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    //基本字节流一次读取一个字节
    public static void method4() {
        FileOutputStream fos = null;
        FileInputStream fis = null;
        try {
            fis = new FileInputStream("D:\\JAVA\\NBA.qlv");
            fos = new FileOutputStream("D:\\JAVA\\JAVAEE\\NBA.qlv");
            int len;
            while ((len = fis.read()) != -1) {
                fos.write(len);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (fos == null || fis == null) {
                try {
                    fos.close();
                    fis.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}


字节缓冲流效率明显高于基本字节流


字符流

为什么会出现字符流

由于字节流操作中文不是特别方便,所以Java就提供了字符流

  • 字符流 = 字节流 + 编码表
    用字节流复制文本文件式,文本文件也会有中文,但是没问题,是因为最终底层操作会自动进行字节拼接成中文,如何识别是中文呢?
  • 汉字在存储的时候,无论选择哪种编码存储,第一个字节都是负数。

编码:

  • byte[] getBytes():使用平台默认字符集将该 String 编码为一系列字节,将结果存储到新的字节数组中
  • byte[] getBytes(String charsetName):使用指定的字符集将该 String编码为一些列字节,将结果存储到新的字节数组中

解码:

  • String(byte[] bytes):通过使用平台的默认字符集解码指定的字节数组来构造新的 String
  • String(byte[] bytes, String charsetName):通过指定的字符集解码指定的字节数组来构造新的String

字符流编码解码问题

  • 字符流抽象基类
    • Reader:字符输入流的抽象类
    • Writer:字符输出流的抽象类
  • 字符流中和编码解码问题相关的两个类
    • InputStreamReader:是从字节流到字符流的桥梁:它读取字节,并使用指定的charset将其解码为字符。 它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集。
    • OutputStreamWriter:是从字符流到字节流的桥梁:使用指定的charset将写入的字符编码为字节。 它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集。
import java.io.*;
public class Demo {
    public static void main(String[] args) throws IOException {
        //编码
//        OutputStreamWriter​(OutputStream out) 创建一个使用默认字符编码的OutputStreamWriter。
//        OutputStreamWriter​(OutputStream out, String charsetName) 创建一个使用命名字符集的OutputStreamWriter。
        //默认编码
        OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("osw.txt"));
        osw.write("中国");
        osw.close();
        //自定义编码
        OutputStreamWriter osw2 = new OutputStreamWriter(new FileOutputStream("osw2.txt"), "GBK");
        osw2.write("中华人民共和国");
        osw2.close();
        //解码
//        InputStreamReader​(InputStream in) 创建一个使用默认字符集的InputStreamReader。
//        InputStreamReader​(InputStream in, String charsetName) 创建一个使用命名字符集的InputStreamReader。
        //默认编码
        InputStreamReader isr = new InputStreamReader(new FileInputStream("osw.txt"));
        int ch;
        while ((ch = isr.read()) != -1){
            System.out.print((char) ch);
        }
        isr.close();
        System.out.println();
        //自定义编码
        InputStreamReader isr2 = new InputStreamReader(new FileInputStream("osw2.txt"),"GBK");
        int len;
        char [] c = new char[1024];
        while((len = isr2.read(c)) != -1){
            System.out.println(new String(c,0,len));
        }
        isr2.close();
    }
}

字符流写数据的5种方式

  • void write(int c) 写一个字符
  • void write(char[] cbuf)写入一个字符数组
  • void write(char[] cbuf, int off, int len)写入字符数组的一部分
  • void write(String str)写一个字符串
  • void write(String str, int off, int len)写入字符串的一部分
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
public class Demo {
    public static void main(String[] args) throws IOException {
        OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("osw.txt"));
        //void write(int c) 写一个字符
        osw.write(97);
        osw.write(98);

        //void write(char[] cbuf)写入一个字符数组
//        char [] c = {'a','b','c','d'};
//        osw.write(c);

        //void write(char[] cbuf, int off, int len)写入字符数组的一部分
//        char [] c = {'a','b','c','d'};
//        osw.write(c,1,2);

        //void write(String str)写一个字符串
//        osw.write("hello");

        //void write(String str, int off, int len)写入字符串的一部分
//        osw.write("hello",1,2);

        //void flush()刷新流
        osw.flush();

        osw.close();//关闭流 释放资源 先刷新 后释放
    }
}

字符流读数据的两种方式

  • int read()一次读一个字符数据
  • int read(char[] cbuf)一次读一个字符数组数据

```import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStreamReader;

public class Demo {
    public static void main(String[] args) throws IOException {
        //int read()一次读一个字符数据
        InputStreamReader isr = new InputStreamReader(new FileInputStream("osw.txt"));
        int ch;
        while ((ch = isr.read()) != -1){
            System.out.print((char)ch);
        }
        isr.close();
        System.out.println("--------------");

        //int read(char[] cbuf)一次读一个字符数组数据
        InputStreamReader isr2 = new InputStreamReader(new FileInputStream("osw.txt"));
        int ch2;
        char [] c = new char[1024];
        while((ch2 = isr2.read(c)) != -1){
            System.out.print(new String(c,0,ch2));
        }
        isr2.close();
    }
}

转换流的名字较长,而我们常见的操作都是按照本地默认编码实现的,为了简便书写,转换流提供了对应的子类

  • FileReader:用于读取取字符的便捷类
    • FileReader(String fileName)
  • FileWriter:用于写入字符文件的便捷类
    • FileWriter(String fileName)

字符缓冲流

  • BufferedReader:从字符输入流读取文本,缓冲字符,以提供字符,数组和行的高效读取。 可以指定缓冲区大小,或者可以使用默认大小。 默认值足够大,可用于大多数用途。
  • BufferedWriter:将文本写入字符输出流,缓冲字符,以提供单个字符,数组和字符串的高效写入。 可以指定缓冲区大小,或者可以接受默认大小。 默认值足够大,可用于大多数用途。

构造方法:

  • BufferedReader(Reader in)
  • BufferedWriter(Writer out)
import java.io.*;
public class Demo {
    public static void main(String[] args) throws IOException {
//        BufferedWriter bw = new BufferedWriter(new FileWriter("bw.txt"));
//        bw.write("hello\r\n");
//        bw.write("world");
//        bw.close();
        BufferedReader br = new BufferedReader(new FileReader("bw.txt"));
        //一次读取一个字符数据
//        int len;
//        while((len = br.read()) != -1){
//            System.out.print((char)len);
//        }
        
        //一次读取一个字符数组
        int len;
        char [] c = new char[1024];
        while((len = br.read(c)) != -1){
            System.out.print(new String(c,0,len));
        }
        br.close();
    }
}

字符缓冲流特有功能

  • BufferedWriter
    • void newLine():写一行行分隔符,行分隔符字符串由系统属性定义
  • BufferedReader:
    • public String readLine():读一行文字。结果包含行的内容的字符串,不包括任何行终止符,如果流的结尾已经到达,则为null
import java.io.*;
public class Demo {
    public static void main(String[] args) throws IOException {
   /*     BufferedWriter bw = new BufferedWriter(new FileWriter("bw.txt"));
        for (int i = 0; i < 10; i++){
            bw.write("hello"+i);
            bw.newLine();
        }
        bw.close();*/

        BufferedReader br = new BufferedReader(new FileReader("bw.txt"));
        String line;
        while((line = br.readLine()) != null){
            System.out.println(line);
        }
        br.close();
    }
}

JDK7之后的异常处理

格式:

try(定义流对象){
    可能出现异常的代码;
}catch(异常类 名变量名){
    异常的处理代码;
}

自动释放资源

JDK9之后的异常处理

定义输入流对象;
定义输出流对象;
try(输入流对象;输出流对象){
 可能出现异常的代码;
}catch(异常类 名变量名){
    异常的处理代码;
}

自动释放资源
例如:

package 复制文件异常加入异常处理;

import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;

public class Demo {
    public static void main(String[] args) {
    }
    
    //JDK9的改进
    private static void method3() throws IOException{
        FileReader fr = new FileReader("fr.txt");
        FileWriter fw = new FileWriter("fw.txt");
        try(fr;fw){
            int len;
            char [] chars = new char[1024];
            while ((len = fr.read(chars)) != -1){
                fw.write(chars,0,len);
            }
        }catch (IOException e){
            e.printStackTrace();
        }
    }

    //JDK7的改进
    private static void method2(){
        try(FileReader fr = new FileReader("fr.txt");
            FileWriter fw = new FileWriter("fw.txt")){
            int len;
            char [] chars = new char[1024];
            while ((len = fr.read(chars)) != -1){
                fw.write(chars,0,len);
            }
        }catch (IOException e){
            e.printStackTrace();
        }
    }

    //try...catch...finally
    private static void method01(){
        FileReader fr = null;
        FileWriter fw = null;
        try{
            fr = new FileReader("fr.txt");
            fw = new FileWriter("fw.txt");
            int len;
            char [] chars = new char[1024];
            while ((len = fr.read(chars)) != -1){
                fw.write(chars,0,len);
            }
        }catch (IOException e){
            e.printStackTrace();
        }finally {
            if (fr != null){
                try{
                    fr.close();
                }catch (IOException e){
                    e.printStackTrace();
                }
            }
        }
        if (fw != null){
            try{
                fw.close();
            }catch (IOException e){
                e.printStackTrace();
            }
        }
    }
}


对象序列化流

对象序列化:就是将对象保存到磁盘中,或网络传输对象
这种机制就是使用一个字节序列表示一个对象,该字节序列包含:对象的类型、对象的数据和对象的存储的属性等信息字节序列写到文件之后,相当于文件中保存了一个对象的信息,反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化

  • 对象序列化流:ObjectOutputStream
  • 对象反序列化流:ObjectInputStream

对象序列化流

ObjectOutputStream将Java对象的原始数据类型和图形写入OutputStream。 可以使用ObjectInputStream读取(重构)对象。 可以通过使用流的文件来实现对象的持久存储。 如果流是网络套接字流,则可以在另一个主机上或另一个进程中重构对象。

构造方法:

  • ObjectOutputStream(OutputStream out):创建一个写入指定的OutputStream的ObjiectOutputStream

序列化对象的方法:

  • void writeObject(Object obj):将指定的对象写入ObjectOutputStream

注意:对象所属的类需要实现Serializable接口.不实现此接口将不会使任何状态序列化或反序列化(仅是一个标识接口,无需要重新的方法)

对象反序列化流

ObjectInputStream反序列化先前使用ObjectOutputStream编写的原始数据和对象。

构造方法:

  • ObjectInputStream(InputStream in) 创建从指定的InputStream读取的ObjectInputStream。

反序列化对象的方法:

  • Object readObject():从ObjectInputStream读取一个对象
  1. 用 transient 关键字修饰的成员变量不会被序列化
  2. 用对象序列流序列化了一个对象后,假如我们修改了对象所属的类文件,读取数据会抛出:InvalidClassException异常.我们可以给对象所属的类加一个serialVersionUID来解决这个问题。
    private static final long serialVersionUID = 42L;
import java.io.Serializable;

public class Student implements Serializable {
    private static final long serialVersionUID = 42L;
    private String name;
    private int age;

    public Student() {
    }

    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}


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

public class Demo {
    public static void main(String[] args) {
        try(ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("oos.txt"))
            ObjectInputStream ois = new  ObjectInputStream(new FileInputStream("oos.txt"))){

            Student student = new Student("张三",20);

            oos.writeObject(student);

            Object obj = ois.readObject();

            Student s = (Student)obj;

            System.out.println(s.getName() + "," + s.getAge());
            
        }catch (IOException | ClassNotFoundException e){
            e.printStackTrace();
        }
    }
}


Properties

  • 是一个Map体系的集合类
  • Properties可以保存到流中或从流中加载
mport java.util.Properties;
import java.util.Set;


public class Demo {
    public static void main(String[] args) {
        Properties p = new Properties();
        p.put("张三",20);
        p.put("李四",18);
        p.put("王五",19);

        Set<Object> keys = p.keySet();
        for (Object key : keys){
            System.out.println(p.get(key));
        }
    }
}

特有方法:

  • Objiec setProperty(String key,String value) 设值集合的键和值,都是String类型,底层调用Hashtable方法put
  • String getProperty(String key) 使用此属性列表中指定的键搜索属性
  • Set stringPropertyNames() 从该属性列表返回一个不可修改的键集,其中键及其对应的值是字符串
import java.util.Properties;
import java.util.Set;

public class Demo2 {
    public static void main(String[] args) {
        //Objiec setProperty(String key,String value)
        Properties p = new Properties();
        p.setProperty("张三", "19");
        p.setProperty("李四", "20");
        p.setProperty("王五", "18");
        System.out.println(p);
        System.out.println("--------------");
        //String getProperty(String key)
        String zs = p.getProperty("张三");
        System.out.println(zs);
        System.out.println("--------------");
        //Set<String> stringPropertyNames()
        Set<String> keys = p.stringPropertyNames();
        for (String key : keys) {
            System.out.println(key);
        }

        System.out.println("------------------");
        Set<String> names = p.stringPropertyNames();
        for (String key : names) {
            String values = p.getProperty(key);
            System.out.println(key + "," + values);
        }
    }
}


Properties与IO流

  1. void load(InputStream inStream) 从输入字节流读取属性列表(键和元素对)
  2. void load(Reader reader)从输入字符流读取属性列表(键和元素对)
  3. void store(OutputStream out,String comments)将此属性列表(键和元素对)写入此Properties表中,以适合于使用load(InputStream)方法的格式写入输出字节流
  4. void store(Writer writer, String comments)将此属性列表(键和元素对)写入此Properties表中,以适合使用load(Reader)方法的格式写入输出字符流
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.util.Properties;
public class Demo {
    public static void main(String[] args) throws IOException{
//        mystore();
        myload();
    }


    private static void mystore() throws IOException {
        Properties p = new Properties();
        p.setProperty("张三","20");
        p.setProperty("李四","19");
        p.setProperty("王五","18");
        //void store(OutputStream out,String comments)   comments描述
        FileWriter fw = new FileWriter("IO流\\fw.txt");
        p.store(fw,null);
        fw.close();

    }

    private static void myload() throws IOException{
        Properties p = new Properties();
        FileReader fr = new FileReader("IO流\\fw.txt");
        p.load(fr);
        System.out.println(p);
    }
}
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!