文件I/O流

一、File

1. File概述

        File是文件和目录路径名的抽象表示,File封装的并不是一个真正存在的文件。仅仅是一个路径名,它可以存在,也可以不存在。后面需要通过对应的方法将其变成真正存在的。

2. 构造方法

方法说明
File(String pathName)根据字符串路径创建一个 File 对象
File(String parent,String child)根据字符串父级路径和子级路径创建 File 对象
File(File parent,String child)根据 File 类型父级路径和字符串子级路径创建 File 对象
//File(String pathName)
File f1 = new File("D:\\src\\java.txt");

//File(String parent,String child)
File f2 = new File("D:\\src", "java.txt");

//File(File parent,String child)
File f3 = new File("D:\\src");
File f4 = new File(f3, "java.txt");

3. 创建功能方法

方法说明
boolean createNewFile()当具有该名称的文件不存在时,创建一个由该抽象路径名命名的新空文件
boolean mkdir()创建由此抽象路径名命名的目录
boolean mkdirs()创建由此抽象路径名命名的目录,包括任何必需但不存在的父目录
//在E:\\itcast目录下创建一个文件java.txt
File f1 = new File("E:\\itcast\\java.txt");
System.out.println(f1.createNewFile());//true

//在E:\\itcast目录下创建一个目录JavaSE
File f2 = new File("E:\\itcast\\JavaSE");
System.out.println(f2.mkdir());//true

//在E:\\itcast目录下创建一个多级目录JavaWEB\\HTML
File f3 = new File("E:\\itcast\\JavaWEB\\HTML");
System.out.println(f3.mkdirs());//true

4. 判断功能方法

方法说明
boolean isDirectory()测试此抽象路径名表示的File是否为目录
boolean isFile()测试此抽象路径名表示的File是否为文件
boolean exists()测试此抽象路径名表示的File是否存在
//创建一个File对象
File f = new File("myFile\\java.txt");

//public boolean isDirectory():测试此抽象路径名表示的File是否为目录
System.out.println(f.isDirectory());
//public boolean isFile():测试此抽象路径名表示的File是否为文件
System.out.println(f.isFile());
//public boolean exists():测试此抽象路径名表示的File是否存在
System.out.println(f.exists());

5. 获取功能方法

方法说明
String getAbsolutePath()返回此抽象路径名的绝对路径名字符串
String getPath()将此抽象路径名转换为路径名字符串
String getName()返回由此抽象路径名表示的文件或目录的名称
String[] list()返回由此抽象路径名表示的目录中的文件和目录的名字字符串数组
File[] listFiles()返回由此抽象路径名表示的目录中的文件和目录的File对象数组
//创建一个File对象
File f = new File("myFile\\java.txt");

//public String getAbsolutePath():返回此抽象路径名的绝对路径名字符串
//public String getPath():将此抽象路径名转换为路径名字符串
//public String getName():返回由此抽象路径名表示的文件或目录的名称
System.out.println(f.getAbsolutePath());
System.out.println(f.getPath());
System.out.println(f.getName());

//public String[] list():返回此抽象路径名表示的目录中的文件和目录的名称字符串数组
//public File[] listFiles():返回此抽象路径名表示的目录中的文件和目录的File对象数组
File f2 = new File("E:\\itcast");

String[] strArray = f2.list();
for(String str : strArray) {
    System.out.println(str);
}

File[] fileArray = f2.listFiles();
for(File file : fileArray) {
    //System.out.println(file);
    //System.out.println(file.getName());
    if(file.isFile()) {
    System.out.println(file.getName());
}

6. 删除功能方法

方法说明
boolean delete()删除由此抽象路径名表示的文件或目录(目录必须为空才可删除)
//在当前模块目录下创建java.txt文件
File f1 = new File("myFile\\java.txt");
System.out.println(f1.createNewFile());//true

//删除当前模块目录下的java.txt文件
System.out.println(f1.delete());//true

//在当前模块目录下创建itcast目录
File f2 = new File("myFile\\itcast");
System.out.println(f2.mkdir());//true

//删除当前模块目录下的itcast目录
System.out.println(f2.delete());//true

//在当前模块下创建一个目录itcast,然后在该目录下创建一个文件java.txt
File f3 = new File("myFile\\itcast");
System.out.println(f3.mkdir());//true
File f4 = new File("myFile\\itcast\\java.txt");
System.out.println(f4.createNewFile());//true

//删除当前模块下的目录itcast
System.out.println(f4.delete());//true
System.out.println(f3.delete());//true

7. 递归

         递归指的是方法本身自己调用自己,把一个复制问题层层转化为一个与原问题相似的规模较小的问题求解。递归策略只需少量的程序就可以描述出解题过程所需要的多次重复计算

        注意:
        递归必须要有出口,递归的次数不宜过多,否则都会造成内存溢出
        递归规则,与原问题相似的规模较小的问题

public class Demo {
    public static void main(String[] args) {
        //兔子问题,求第20个月兔子的对数,每个月的兔子对数:1,1,2,3,5,8,...
        int[] arr = new int[20];
        arr[0] = 1;
        arr[1] = 1;
        for (int i = 2; i < arr.length; i++) {
            arr[i] = arr[i - 1] + arr[i - 2];
        }
        System.out.println(arr[19]);
        System.out.println(f(20));
    }
    /*
        递归解决问题,首先就是要定义一个方法:
            定义一个方法f(n):表示第n个月的兔子对数
            那么,第n-1个月的兔子对数该如何表示呢?f(n-1)
            同理,第n-2个月的兔子对数该如何表示呢?f(n-2)
        StackOverflowError:当堆栈溢出发生时抛出一个应用程序递归太深
     */
    public static int f(int n) {
        if(n==1 || n==2) {
            return 1;
        } else {
            return f(n - 1) + f(n - 2);
        }
    }
}

二、I/O流

1. 概述

        IO:输入/输出(Input/Output)

        流:是一种抽象概念,是对数据传输的总称。也就是说哦数据在设备之间的传输称为流,流的本质是数据传输

        IO流就是用来处理设备间数据传输问题的,常见应用于文件复制、文件上传、文件下载

       输入(读数据):硬盘→内存        输出(写数据):内存→硬盘

        按数据类型分为字节流(字节输入流和字节输出流)和字符流(字符输入流和字符输出流)

2. 字节流

        子类名特点:以其父类名作为子类名的后缀

2.1 InputStream字节输入流

        InputStream抽象类表示字节输入流的所有超类

        FileInputStream是文件输入流用于从File读取数据

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

使用字节输入流读数据步骤:

        ①创建字节输入流对象

        ②调用字节输入流的对象读数据方法

        ③释放资源

2.1.1 字节输入流读数据的两种方式

方法说明
int read()将指定的字节读取此文件输入流,一次读取一个字节数据
int read(byte[] b)将b.length字节从指定的字节数组读取此文件输入流,一次读取一个字节数组
public class FileInputStreamDemo01 {
    public static void main(String[] args) throws IOException {
        //创建字节输入流对象
        //FileInputStream(String name)
        FileInputStream fis = new FileInputStream("myByteStream\\fos.txt");

        //调用字节输入流对象的读数据方法
        //int read():从该输入流读取一个字节的数据
        /*
        //第一次读取数据
        int by = fis.read();
        System.out.println(by);
        System.out.println((char)by);

        //第二次读取数据
        by = fis.read();
        System.out.println(by);
        System.out.println((char)by);

        //再多读取两次
        by = fis.read();
        System.out.println(by);
        by = fis.read();
        System.out.println(by);

        //如果达到文件的末尾, -1
        */
        /*
        int by = fis.read();
        while (by != -1) {
            System.out.print((char)by);
            by = fis.read();
        }
        */
        //优化上面的程序
        int by;
        /*
            fis.read():读数据
            by=fis.read():把读取到的数据赋值给by
            by != -1:判断读取到的数据是否是-1
         */
        while ((by=fis.read())!=-1) {
            System.out.print((char)by);
        }


        //调用字节输入流对象的读数据方法
        //int read​(byte[] b):从该输入流读取最多 b.length个字节的数据到一个字节数组
        /*
        byte[] bys = new byte[5];

        //第一次读取数据
        int len = fis.read(bys);
        System.out.println(len);
        //String​(byte[] bytes)
//        System.out.println(new String(bys));
        System.out.println(new String(bys,0,len));

        //第二次读取数据
        len = fis.read(bys);
        System.out.println(len);
//        System.out.println(new String(bys));
        System.out.println(new String(bys,0,len));

        //第三次读取数据
        len = fis.read(bys);
        System.out.println(len);
        //String​(byte[] bytes, int offset, int length)
        System.out.println(new String(bys,0,len));

        //再多读取两次
        len = fis.read(bys);
        System.out.println(len);
        len = fis.read(bys);
        System.out.println(len);
        */

        /*
            hello\r\n
            world\r\n

            第一次:hello
            第二次:\r\nwor
            第三次:ld\r\nr

         */
        byte[] bys = new byte[1024]; //1024及其整数倍
        int len;
        while ((len=fis.read(bys))!=-1) {
            System.out.print(new String(bys,0,len));
        }

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

2.2 OutputStream字节输出流

        OutputStream抽象类表示字节输出流的所有超类

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

方法说明
public FileOutputStream(String name)创建文件输出流以指定的名称写入文件
public FileOutputStream(String name, boolean append)创建文件输出流以指定的名称写入文件,如果append为 true,则字节将写入文件的末尾

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

        ①创建字节输出流对象(调用系统功能创建了文件,创建了字节输出流对象,让字节输出流对象指向创建好的文件)

        ②调用字节输出流对象的写数据方法

        ③释放资源(关闭此文件输出流并释放与此相关联的任何系统资源)

 字节流写数据实现换行:

        Windows:\r\n        linux::\n        mac:\r

public class FileOutputStreamDemo01 {
    public static void main(String[] args) throws IOException {
        //创建字节输出流对象
        //FileOutputStream​(String name):创建文件输出流以指定的名称写入文件
        FileOutputStream fos = new FileOutputStream("myByteStream\\fos.txt");
        /*
            做了三件事情:
                A:调用系统功能创建了文件
                B:创建了字节输出流对象
                C:让字节输出流对象指向创建好的文件
         */
        //void write​(int b):将指定的字节写入此文件输出流
        fos.write(97);//a

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

2.2.1 字节输出流写数据的三种方式

方法说明
void write(int b)将指定的字节写入此文件输出流,一次写一个字节数据
void write(byte[] b)将 b.length字节从指定的字节数组写入此文件输出流,一次写一个字节数组
void write(byte[] b, int off, int len)将len字节从指定字节数组开始,从偏移量off开始写入此文件输出流,一次写一个字节数组的部分数据
public class FileOutputStreamDemo02 {
    public static void main(String[] args) throws IOException {
        //FileOutputStream​(String name):创建文件输出流以指定的名称写入文件
        FileOutputStream fos = new FileOutputStream("myByteStream\\fos.txt");
        //new File(name)
//        FileOutputStream fos = new FileOutputStream(new File("myByteStream\\fos.txt"));

        //FileOutputStream​(File file):创建文件输出流以写入由指定的 File对象表示的文件
//        File file = new File("myByteStream\\fos.txt");
//        FileOutputStream fos2 = new FileOutputStream(file);
//        FileOutputStream fos2 = new FileOutputStream(new File("myByteStream\\fos.txt"));

        //void write​(int b):将指定的字节写入此文件输出流
//        fos.write(97);

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

        //void write​(byte[] b, int off, int len):将 len字节从指定的字节数组开始,从偏移量off开始写入此文件输出流
//        fos.write(bys,0,bys.length);
        fos.write(bys,1,3);

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

2.3 字节缓冲流

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

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

构造方法说明
BufferedInputStream​(InputStream in)创建一个BufferedInputStream并保存其参数,输入流in供以后使用
BufferedOutputStream​(OutputStream out)创建一个新的缓冲输出流,以将数据写入指定的底层输出流
public class BufferStreamDemo {
    public static void main(String[] args) throws IOException {
        //字节缓冲输出流:BufferedOutputStream​(OutputStream out)
//        FileOutputStream fos = new FileOutputStream("myByteStream\\bos.txt");
//        BufferedOutputStream bos = new BufferedOutputStream(fos);
        /*
        BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("myByteStream\\bos.txt"));
        //写数据
        bos.write("hello\r\n".getBytes());
        bos.write("world\r\n".getBytes());
        //释放资源
        bos.close();
        */

        //字节缓冲输入流:BufferedInputStream​(InputStream in)
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream("myByteStream\\bos.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.print(new String(bys,0,len));
        }

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

3. 字符流

        字符流 = 字节流 + 编码表
        无论汉字存储时选择那种编码格式,第一个字节都是负数

3.1 编码表

        计算机中储存的信息都是用二进制数表示的;我们在屏幕看到的英文、汉字都是二进制数转换后的结果。按照某种规则,将字符存储到计算机中,称为编码。反之,将存储在计算机中的二进制数按照某种规则解析显示出来,称为解码

        字符编码:就是一套自然语言的字符与二进制数之间的对应规则,如:(A, 65)

常见的编码格式:
        ASCII(American Standard Code for Information Interchange,美国信息交换标准代码):是基于拉丁字母的一套电脑编码系统,用于显示现代英语
        GBK:最常用的中文码表。是在 GB2312 标准基础上的扩展规范,使用了双字节编码方案
        UTF-8编码:可以用来表示Unicode标准中任意字符,大部分常用字(含中文),使用三个字节编码

注意:
        采用何种规则编码,就要采用对应规则解码,否则就会乱码

3.2 字符串编解码

编码

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

解码

方法说明
String​(byte[] bytes)通过使用平台的默认字符集解码指定的字节数组来构造新的字符串
String​(byte[] bytes, String charsetName)通过指定的字符集解码指定的字节数组来构造新的字符串
public class StringDemo {
    public static void main(String[] args) throws UnsupportedEncodingException {
        //定义一个字符串
        String s = "中国";

        //byte[] getBytes():使用平台的默认字符集将该 String编码为一系列字节,将结果存储到新的字节数组中
        //byte[] bys = s.getBytes(); //[-28, -72, -83, -27, -101, -67]
        //byte[] getBytes(String charsetName):使用指定的字符集将该 String编码为一系列字节,将结果存储到新的字节数组中
//        byte[] bys = s.getBytes("UTF-8"); //[-28, -72, -83, -27, -101, -67]
        byte[] bys = s.getBytes("GBK"); //[-42, -48, -71, -6]
        System.out.println(Arrays.toString(bys));

        //String(byte[] bytes):通过使用平台的默认字符集解码指定的字节数组来构造新的 String
//        String ss = new String(bys);
        //String(byte[] bytes, String charsetName):通过指定的字符集解码指定的字节数组来构造新的 String
//        String ss = new String(bys,"UTF-8");
        String ss = new String(bys,"GBK");
        System.out.println(ss);
    }
}

3.3 字符流编解码(转换流)

        Reader字符输入流的抽象类,Writer字符输出流的抽象类

        字符流中与编解码相关的两个类:InputStreamReader与OutputStreamWriter

构造方法说明
InputStreamReader(InputStream in)创建一个使用默认字符编码的InputStreamReader
InputStreamReader(InputStream in, String charSetName)创建一个使用命名字符集的InputStreamReader
OutputStreamWriter(OutputStream out)创建一个使用默认字符编码的OutputStreamWriter
OutputStreamWriter(OutputStream out, String charSetName)创建一个使用命名字符集的OutputStreamWriter
public class ConversionStreamDemo {
    public static void main(String[] args) throws IOException {
//        OutputStreamWriter​(OutputStream out) 创建一个使用默认字符编码的OutputStreamWriter。
//        OutputStreamWriter​(OutputStream out, String charsetName) 创建一个使用命名字符集的OutputStreamWriter。
//        FileOutputStream fos = new FileOutputStream("myCharStream\\osw.txt");
//        OutputStreamWriter osw = new OutputStreamWriter(fos);
//        OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("myCharStream\\osw.txt"));
//        OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("myCharStream\\osw.txt"),"UTF-8");
        OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("myCharStream\\osw.txt"),"GBK");
        osw.write("中国");
        osw.close();

//        InputStreamReader​(InputStream in) 创建一个使用默认字符集的InputStreamReader。
//        InputStreamReader​(InputStream in, String charsetName) 创建一个使用命名字符集的InputStreamReader。
//        InputStreamReader isr = new InputStreamReader(new FileInputStream("myCharStream\\osw.txt"));
        InputStreamReader isr = new InputStreamReader(new FileInputStream("myCharStream\\osw.txt"),"GBK");
        //一次读取一个字符数据
        int ch;
        while ((ch=isr.read())!=-1) {
            System.out.print((char)ch);
        }
        isr.close();
    }
}

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

构造方法说明
FileReader(String fileName)用于读取字符文件的便捷类
FileWriter(String fileName)用于写入字符文件的便捷类

3.4 写数据方法

方法名说明
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)写一个字符串的一部分

        void flush()方法: 刷新流,还能继续写数据

public class OutputStreamWriterDemo {
    public static void main(String[] args) throws IOException {
        //OutputStreamWriter​(OutputStream out):创建一个使用默认字符编码的OutputStreamWriter
        OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("myCharStream\\osw.txt"));
        //void write​(int c):写一个字符
//        osw.write(97);
//        //void flush():刷新流
//        osw.flush();
//        osw.write(98);
//        osw.flush();
//        osw.write(99);
        //void write​(char[] cbuf):写入一个字符数组
        char[] chs = {'a', 'b', 'c', 'd', 'e'};
//        osw.write(chs);
        //void write​(char[] cbuf, int off, int len):写入字符数组的一部分
//        osw.write(chs, 0, chs.length);
//        osw.write(chs, 1, 3);
        //void write​(String str):写一个字符串
//        osw.write("abcde");
        //void write​(String str, int off, int len):写一个字符串的一部分
//        osw.write("abcde", 0, "abcde".length());
        osw.write("abcde", 1, 3);
        //释放资源
        osw.close();
        //Exception in thread "main" java.io.IOException: Stream closed
//        osw.write(100);
    }
}

3.5 读数据方法

方法名说明
int read()一次读一个字符数据
int read(char[] cbuf)一次读一个字符数组数据
public class InputStreamReaderDemo {
    public static void main(String[] args) throws IOException {
        //InputStreamReader​(InputStream in):创建一个使用默认字符集的InputStreamReader
//        InputStreamReader isr = new InputStreamReader(new FileInputStream("myCharStream\\osw.txt"));
        InputStreamReader isr = new InputStreamReader(new FileInputStream("myCharStream\\ConversionStreamDemo.java"));
        //int read​():一次读一个字符数据
//        int ch;
//        while ((ch=isr.read())!=-1) {
//            System.out.print((char)ch);
//        }
        //int read​(char[] cbuf):一次读一个字符数组数据
        char[] chs = new char[1024];
        int len;
        while ((len = isr.read(chs)) != -1) {
            System.out.print(new String(chs, 0, len));
        }
        //释放资源
        isr.close();
    }
}

3.6 字符缓冲流

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

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

特有方法说明
void newLine()BufferedWriter特有方法,写一行行分隔符,行分隔符字符串由系统属性定义
String readLine()BufferedReader特有方法,读一行文字。 结果包含行的内容的字符串,不包括任何行终止字符,如果流的结尾已经到达,则为 null

4. 异常处理

4.1 try...catch...finally

private static void method1() {
    FileReader fr = null;
    FileWriter fw = null;
    try {
        fr = new FileReader("fr.txt");
        fw = new FileWriter("fw.txt");

        char[] chs = new char[1024];
        int len;
        while ((len = fr.read()) != -1) {
            fw.write(chs, 0, len);
        }
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        if(fw!=null) {
            try {
                fw.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
        if(fr!=null) {
            try {
                fr.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

4.2 JDK7改进方案

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

4.3 JDK9改进方案

//需要抛出异常
private static void method4() throws IOException {
    FileReader fr = new FileReader("fr.txt");
    FileWriter fw = new FileWriter("fw.txt");
    try (fr; fw) {
        char[] chs = new char[1024];
        int len;
        while ((len = fr.read()) != -1) {
            fw.write(chs, 0, len);
        }
    } catch (IOException e) {
        e.printStackTrace();
    }
}

三、特殊操作流

1. 标准输入输出流

        public final InputStream in:通常该流对应于键盘输入或由主机环境或用户指定的另一个输入源

        我们可以自己实现键盘录入输入。
        new BufferedReader(new InputStreamReader(System.in));调用readLine()方法接收数据

        但是上面的方式太麻烦了,所以我们直接使用键盘输入的类
        Scanner sc = new Scanner(System.in);

        public final OutputStream out:通常该流对应于显示输出或由主机环境或用户指定的另一个输出目标

        输出语句本质就是一个标准输出流

使用步骤:
        ①创建标准输出流对象:PrintStream ps = System.out;
        ②调用print()方法可以输出不带换行效果的数据
        ③调用println()方法可以输出带换行效果的数据
        ④PrintStream 类拥有的方法,System.out都可以使用

2. 打印流

        PrintStream:字节打印流,只负责输出数据,不负责读取数据;有自己特有方法

使用步骤:

        ①创建字节打印流对象  PrintStream ps =new PrintStream(String fileName);
        ②调用字节流写出数据的方法:
        ③write()写数据。会根据编码转换
        ④调用print()方法写出数据,不会进行转码,不会换行
        ⑤调用println()方法写出数据,不会进行转码,会换行

        PrintWriter:字符打印流

构造方法说明
PrintWriter(String fileName)使用指定的文件名创建一个新的 PrintWriter,而不需要自动执行刷新
PrintWriter(Writer out, boolean autoFlush)创建一个新的 PrintWriter,out:字符输出流,autoFlush:一个布尔值,若为真,则 println,printf,或 format 方法将刷新输出缓冲区

使用步骤:
        ①创建字符打印流对象: PrintWriter ps = new PrintWrtier(文件路径);
        ②调用write()方法,写出数据。并写出换行。
        ③调用flush()方法,将数据刷新到文件中。
        ④创建字符打印流对象:PrintWriter pw = new PrintWriter(文件路径,true);
        ⑤调用println()方法,写出数据

3. 对象序列化流

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

要实现序列化和反序列化就要使用:
        对象序列化流:ObjectOutputStream        对象反序列化流:ObjectInputStream

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

构造方法:

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

序列化对象的方法:
        void writeObject(Object obj):将指定的对象写入 ObjectOutputStream

注意:
        一个对象要想被序列化,该对象所属的类必须实现Serializable接口,Serializable是一个标记接口,实现该接口不需重写任何方法

        serialVersionUID:序列化号,可以解决类文件不一致的问题,强烈建议所有可序列化的类显式声明serialVersionUID值,并使用private修饰符

        transient:如果一个成员不想参与序列化操作,可以使用该关键字

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

构造方法:

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

反序列化对象的方法:
        Object readObject():从 ObjectInputStream 读取一个对象

4. Properties

        Properties:是一个 Map 体系的集合类,可以保存到流中或从流中加载

4.1 特有方法

方法名说明
Object setProperty​(String key, String value)设置集合的键和值,都是String类型,底层调用Hashtable方法 put
String getProperty​(String key)使用此属性列表中指定的键搜索属性
Set<String> stringPropertyNames​()从该属性列表中返回一个不可修改的键集,其中键及其对应的值是字符串,包括默认属性列表中的不同键,如果尚未从主属性列表中找到相同名称的键

4.2 与I/O流结合方法

方法名说明
void load(InputStream inStream)从输入字节流读取属性列表(键和元素对)
void load(Reader reader)从输入字符流读取属性列表(键和元素对)
void store(OutputStream out, String comments)将此属性列表(键和元素对)写入此 Properties 表中,以适合于使用 load(InputStream) 方法的格式写入输出字节流
void store(Writer writer, String comments)将此属性列表(键和元素对)写入此 Properties 表中,以适合于使用 load(Reader) 方法的格式写入输出字符流

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值