【JAVA入门】IO流

发布于:2023-01-04 ⋅ 阅读:(529) ⋅ 点赞:(0)

目录

思维导图

一、File类

1、File类概述和构造方法

二、递归

1、递归

递归遍历目录

三、IO流

1、什么是IO流

2、IO流的分类

3、四大IO流

4、Java要掌握的流(16个)

5、字节流(Input/Output)

6、字符流(Reader/Writer)

字符串中的编码解码问题

字符流中的编码解码问题

字符流写数据的5种方式

1.6字符流读数据的2种方式【应用】

字符缓冲流【应用】

字符缓冲流特有功能

7、特殊流

Properties集合

Properties作为Map集合的使用

Properties作为Map集合的特有方法

Properties和IO流相结合的方法


思维导图

 

一、File类

1、File类概述和构造方法

File类介绍

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

常用方法汇总

方法名 说明
File(String pathname) 通过将给定的路径名字符串转换为抽象路径名来创建新的 File实例
File(String parent, String child) 从父路径名字符串和子路径名字符串创建新的 File实例
File(File parent, String child) 从父抽象路径名和子路径名字符串创建新的 File实例
方法名 说明
public boolean createNewFile() 当具有该名称的文件不存在时,创建一个由该抽象路径名命名的新空文件
public boolean mkdir() 创建由此抽象路径名命名的目录
public boolean mkdirs() 创建由此抽象路径名命名的目录,包括任何必需但不存在的父目录
方法名 说明
public boolean isDirectory() 测试此抽象路径名表示的File是否为目录
public boolean isFile() 测试此抽象路径名表示的File是否为文件
public boolean exists() 测试此抽象路径名表示的File是否存在
方法名 说明
public String getAbsolutePath() 返回此抽象路径名的绝对路径名字符串
public String getPath() 将此抽象路径名转换为路径名字符串
public String getName() 返回由此抽象路径名表示的文件或目录的名称
public String[] list() 返回此抽象路径名表示的目录中的文件和目录的名称字符串数组
public File[] listFiles() 返回此抽象路径名表示的目录中的文件和目录的File对象数组

注意:getName()方法,如果参数是一个文件,则获取的是文件名和后缀 listFiles()方法:进入文件夹,获取这个文件夹里边所有的文件夹和文件,包括隐藏文件夹和文件,并把这些file对象放在一个file类型的数组中返回

listFiles()方法注意:1.当参数是一个文件时,返回null,因为这个方法必须进入文件夹

2.当参数是一个空文件夹时,返回的是一个长度为0的数组

3.当参数是一个有内容的文件夹时,会进入这个文件夹,获取所有的文件和文件夹 并把这些file对象放在一个file类型的数组中返回

4.当参数是一个有权限的文件夹时候,返回的是null

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

注意:删除功能不走回收站 如果删除的是文件,可以直接删除,如果是文件夹,可以删除空文件夹 如果删除的是一个有内容的文件夹,需要先删除文件夹里边的内容,才能删除文件夹 只能删除文件和空文件夹

实例代码:

public class FileDemo01 {
    public static void main(String[] args) {
        //file:文件、目录、不存在的文件或目录
//File(String pathname):通过将给定的路径名字符串转换为抽象路径名来创建新的 File实例。
        File f1 = new File("E:\\itcast\\java.txt");
        System.out.println(f1);
  //文件分割符自适应File.pathSeparator
        File f2 = new File("E:"+File.pathSeparator+"itcast","java.txt");
        System.out.println(f2);
//File(File parent, String child):从父抽象路径名和子路径名字符串创建新的 File实例。
        File f3 = new File("E:\\itcast");
        File f4 = new File(f3,"java.txt");
        System.out.println(f4);
//创建一个与a.txt文件同目录下的另外一个文件b.txt
        File f3=new File("d:\\wyyy\\java1\\a.txt");//创建文件
        f3.createNewFile();//创建文件
        File f4=new File(file1.getParent(),"b.txt");
        System.out.println(f4.createNewFile());
        //需求1:要在E:\\itcast目录下创建一个文件java.txt
        File f1 = new File("E:\\itcast\\java.txt");
        System.out.println(f1.createNewFile());
        System.out.println("--------");
​
        //需求2:我要在E:\\itcast目录下创建一个目录JavaSE
        File f2 = new File("E:\\itcast\\JavaSE");
        System.out.println(f2.mkdir());
        System.out.println("--------");
​
        //需求3:我要在E:\\itcast目录下创建一个多级目录JavaWEB\\HTML
        File f3 = new File("E:\\itcast\\JavaWEB\\HTML");
        System.out.println(f3.mkdirs());
        System.out.println("--------");
​
        //需求4:我要在E:\\itcast目录下创建一个文件javase.txt
        File f4 = new File("E:\\itcast\\javase.txt");
        System.out.println(f4.createNewFile());
        //创建一个File对象
        File f = new File("myFile\\java.txt");
//      public boolean isDirectory():测试此抽象路径名表示的File是否为目录
//      public boolean isFile():测试此抽象路径名表示的File是否为文件
//      public boolean exists():测试此抽象路径名表示的File是否存在
        f.isDirectory();
        f.isFile();
        f.exists();
​
//      public String getAbsolutePath():返回此抽象路径名的绝对路径名字符串
//      public String getPath():将此抽象路径名转换为路径名字符串
//      public String getName():返回由此抽象路径名表示的文件或目录的名称
        f.getAbsolutePath();
        f.getPath();
        f.getName();
        System.out.println("--------");
​
//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);
        }
        System.out.println("--------");
        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());
            }
        }
       //        File f1 = new File("E:\\itcast\\java.txt");
        //需求1:在当前模块目录下创建java.txt文件
        File f1 = new File("myFile\\java.txt");
//        System.out.println(f1.createNewFile());
        //需求2:删除当前模块目录下的java.txt文件
        f1.delete();//彻底删除,不过回收站
        System.out.println("--------");
        //需求3:在当前模块目录下创建itcast目录
        File f2 = new File("myFile\\itcast");
//        System.out.println(f2.mkdir());
        //在当前模块下创建一个目录itcast,然后在该目录下创建一个文件java.txt
        File f3 = new File("myFile\\itcast");
//        System.out.println(f3.mkdir());
        File f4 = new File("myFile\\itcast\\java.txt");
//        System.out.println(f4.createNewFile());
        file.length();//文件大小
   
        
    }
}        
   

 创建目录使用mkdirs就可以,他既可以创建单级目录,也可以创建多级目录,底层代码调用了mkdir。

绝对路径和相对路径的区别

  • 绝对路径:完整的路径名,不需要任何其他信息就可以定位它所表示的文件。例如:E:\itcast\java.txt

  • 相对路径:必须使用取自其他路径名的信息进行解释。例如:myFile\java.txt

  • 如果file(“绝对路径”):getpath()和getAbsolutePath()的结果一样

  • 如果file(“相对路径”):getpath()打印相对路径;getAbsolutePath()打印绝对路径

二、递归

1、递归

递归的介绍

  • 以编程的角度来看,递归指的是方法定义中调用方法本身的现象

  • 把一个复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解

  • 递归策略只需少量的程序就可描述出解题过程所需要的多次重复计算

递归的注意事项

  • 递归一定要有出口。否则内存溢出

  • 递归虽然有出口,但是递归的次数也不宜过多。否则内存溢出

递归遍历目录

  • 案例需求

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

  • 代码实现

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

三、IO流

1、什么是IO流

I : Input O : Output 通过IO可以完成硬盘文件的读和写。

先进先出。

2、IO流的分类

按照 流的方向 进行分类: 以内存作为参照物:

往内存中去:叫做输入(Input)。或者叫做读(Read)。 从内存中出来:叫做输出(Output)。或者叫做写(Write)。 按照 读取数据方式 不同进行分类: 按照 字节 的方式读取数据,一次读取1个字节byte,等同于一次读取8个二进制位。 这种流是万能的,什么类型的文件都可以读取。包括:文本文件,图片,声音文件,视频文件 等…

eg. 假设文件file1.txt,采用字节流的话是这样读的: a中国bc张三fe 第一次读:一个字节,正好读到’a’ 第二次读:一个字节,正好读到’中’字符的一半。 第三次读:一个字节,正好读到’中’字符的另外一半。

按照 字符 的方式读取数据的,一次读取一个字符. 这种流是为了方便读取 普通文本文件 而存在的,这种流不能读取:图片、声音、视频等文件。只能读取 纯文本文件,连word文件都无法读取。

注意:

纯文本文件,不单单是.txt文件,还包括 .java、.ini、.py 。总之只要 能用记事本打开 的文件都是普通文本文件。

eg. 假设文件file1.txt,采用字符流的话是这样读的: a中国bc张三fe 第一次读:'a’字符('a’字符在windows系统中占用1个字节。) 第二次读:'中’字符('中’字符在windows系统中占用2个字节。)

综上所述:流的分类:

输入流、输出流 字节流、字符流

3、四大IO流

字节流

java.io.InputStream 字节输入流 java.io.OutputStream 字节输出流

字符流

java.io.Reader 字符输入流 java.io.Writer 字符输出流 注意:

四大IO流都是抽象类。(abstract class) 所有的流都实现了: java.io.Closeable接口,都是可关闭的,都有 close() 方法。 流是一个管道,这个是内存和硬盘之间的通道,用完之后一定要关闭,不然会耗费(占用)很多资源。 所有的 输出流 都实现了: java.io.Flushable接口,都是可刷新的,都有 flush() 方法。 养成一个好习惯,输出流在最终输出之后,一定要记得flush()刷新一下。这个刷新表示将通道/管道当中剩余未输出的数据强行输出完(清空管道!)刷新的作用就是清空管道。 ps:如果没有flush()可能会导致丢失数据。

在java中只要“类名”以 Stream 结尾的都是字节流。以“ Reader/Writer ”结尾的都是字符流。

IO流的使用场景

  • 如果操作的是纯文本文件,优先使用字符流

  • 如果操作的是图片、视频、音频等二进制文件。优先使用字节流

  • 如果不确定文件类型,优先使用字节流。字节流是万能的流

4、Java要掌握的流(16个)

文件专属: java.io.FileInputStream              java.io.FileOutputStream

                  java.io.FileReader             java.io.FileWriter

转换流:(将字节流转换成字符流) java.io.InputStreamReader         java.io.OutputStreamWriter

缓冲流专属: java.io.BufferedReader           java.io.BufferedWriter                                                            java.io.BufferedInputStream          java.io.BufferedOutputStream

数据流专属: java.io.DataInputStream          java.io.DataOutputStream

标准输出流: java.io.PrintWriter          java.io.PrintStream

对象专属流: java.io.ObjectInputStream                 java.io.ObjectOutputStream

File文件类 :    java.io.File

5、字节流(Input/Output)

字节流抽象基类

  • InputStream:这个抽象类是表示字节输入流的所有类的超类

  • OutputStream:这个抽象类是表示字节输出流的所有类的超类

  • 子类名特点:子类名称都是以其父类名作为子类名的后缀

字节输入流

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

  • 字节输入流读取数据的步骤

    • 创建字节输入流对象

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

    • 释放资源

package io;
import java.io.*;
public class Demo2 {
    public static void main(String[] args) throws IOException {
       InputStream in =new FileInputStream(new File("d:/wy.txt"));
       //读一个字节
        in.read();
        // 读一堆字节
       // byte [] buf = new byte[1024];//预估文件大小,能否装下一堆字节不确定
        byte [] buf = new byte[in.available()];//自动匹配文件大小
        in.read(buf);//将文件内容读取到buf中
        System.out.println(new String(buf));
        //关闭流
        in.close();
    }
}

字节输出流

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

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

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

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

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

public class Demo2 {
    public static void main(String[] args) throws IOException {
       OutputStream out =new FileOutputStream("d:/wy.txt");
       out.write("igaoukjcnlk".getBytes());//将指定内容输出到文件
       out.close();//关闭流
    }

字节流写数据的三种方式

  • 写数据的方法分类

    方法名 说明
    void write(int b) 将指定的字节写入此文件输出流 一次写一个字节数据
    void write(byte[] b) 将 b.length字节从指定的字节数组写入此文件输出流 一次写一个字节数组数据
    void write(byte[] b, int off, int len) 将 len字节从指定的字节数组开始,从偏移量off开始写入此文件输出流 一次写一个字节数组的部分数据

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

  • 字节流写数据如何实现换行

    • windows:\r\n

    • linux:\n

    • mac:\r

  • 字节流写数据如何实现追加写入

    • public FileOutputStream(String name,boolean append)

    • 创建文件输出流以指定的名称写入文件。如果第二个参数为true ,则字节将写入文件的末尾而不是开头

public class FileOutputStreamDemo02 {
    public static void main(String[] args) throws IOException {
//FileOutputStream(String name):创建文件输出流以指定的名称写入文件
FileOutputStream fos = new FileOutputStream("fos.txt");
//new File(name)
FileOutputStream fos = new FileOutputStream(new File("fos.txt"));
//FileOutputStream(File file):创建文件输出流以写入由指定的 File对象表示的文件
 File file = new File("fos.txt");
 FileOutputStream fos2 = new FileOutputStream(file);
 FileOutputStream fos2 = new FileOutputStream(new File("fos.txt"));
//void write(int b):将指定的字节写入此文件输出流
       fos.write(97);
       fos.write(98);
//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();
    }
}

字节流复制文本文件

  • 实现步骤

    • 复制文本文件,其实就把文本文件的内容从一个文件中读取出来(数据源),然后写入到另一个文件中(目的地)

public class Demo2 {
    public static void main(String[] args) throws IOException {
       InputStream in = new FileInputStream("d:/wy.txt");
       OutputStream out = new FileOutputStream("d:/wyy.txt");
       byte[] buf=new byte[10];//如果没东西了会返回-1
       int len;
       while ((len=in.read(buf))!=-1){
          out.write(buf,0,len);//从0开始到len结束
       }
       if(out!=null){out.close();}
       if(in!=null){in.close();}
    }
}

6、字符流(Reader/Writer)

  • 字符流的介绍

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

    字符流 = 字节流 + 编码表

  • 中文的字节存储方式

    用字节流复制文本文件时,文本文件也会有中文,但是没有问题,原因是最终底层操作会自动进行字节拼接成中文,如何识别是中文的呢?

    汉字在存储的时候,无论选择哪种编码存储,第一个字节都是负数

    字符流替换文件内容并输出

package io;
import java.io.*;
public class Demo2 {
    public static void main(String[] args) throws IOException {
      //文件输入到内存
       Reader reader = new FileReader("d:/wy.txt");
       Writer writer=new FileWriter("d:/wyy.txt");
       char[] ch=new char[4];//字符流用char
       StringBuffer sb= new StringBuffer();//用于读取数据的拼接
       int len;
       while((len=reader.read(ch))!=-1){
          sb.append(ch,0,len);
       }
       //在内存中替换占位符
       String str=sb.toString();//将stringBuffered转换为String
       str= str.replace("{name}","wy");
       //输出到文件
       writer.write(str);
       if (writer!=null)writer.close();
       if (reader!=null)reader.close();
    }
}

编码表

  • 什么是字符集

    是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等计算机要准确的存储和识别各种字符集符号,就需要进行字符编码,一套字符集必然至少有一套字符编码。常见字符集有ASCII字符集、GBXXX字符集、Unicode字符集等

  • 常见的字符集

    • ASCII字符集:

      ASCII:是基于拉丁字母的一套电脑编码系统,用于显示现代英语,主要包括控制字符(回车键、退格、换行键等)和可显示字符(英文大小写字符、阿拉伯数字和西文符号)

      基本的ASCII字符集,使用7位表示一个字符,共128字符。ASCII的扩展字符集使用8位表示一个字符,共256字符,方便支持欧洲常用字符。是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等

    • GBXXX字符集

      GBK:最常用的中文码表。是在GB2312标准基础上的扩展规范,使用了双字节编码方案,共收录了21003个汉字,完全兼容GB2312标准,同时支持繁体汉字以及日韩汉字等

    • Unicode字符集:

      UTF-8编码:可以用来表示Unicode标准中任意字符,它是电子邮件、网页及其他存储或传送文字的应用 中,优先采用的编码。互联网工程工作小组(IETF)要求所有互联网协议都必须支持UTF-8编码。它使用一至四个字节为每个字符编码

      编码规则:

      128个US-ASCII字符,只需一个字节编码

      拉丁文等字符,需要二个字节编码

      大部分常用字(含中文),使用三个字节编码

      其他极少使用的Unicode辅助字符,使用四字节编码

字符串中的编码解码问题

  • 相关方法

    方法名 说明
    byte[] getBytes() 使用平台的默认字符集将该 String编码为一系列字节
    byte[] getBytes(String charsetName) 使用指定的字符集将该 String编码为一系列字节
    String(byte[] bytes) 使用平台的默认字符集解码指定的字节数组来创建字符串
    String(byte[] bytes, String charsetName) 通过指定的字符集解码指定的字节数组来创建字符串
public class StringDemo {
    public static void main(String[] args) throws UnsupportedEncodingException {
        //定义一个字符串
        String s = "中国";
//byte[] bys = s.getBytes(); //[-28, -72, -83, -27, -101, -67]
//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 ss = new String(bys);
        //String ss = new String(bys,"UTF-8");
        String ss = new String(bys,"GBK");
        System.out.println(ss);
    }
}

字符流中的编码解码问题

  • 字符流中和编码解码问题相关的两个类

    • InputStreamReader:是从字节流到字符流的桥梁

      它读取字节,并使用指定的编码将其解码为字符

      它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集

    • OutputStreamWriter:是从字符流到字节流的桥梁

      是从字符流到字节流的桥梁,使用指定的编码将写入的字符编码为字节

      它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集

  • 构造方法

    方法名 说明
    InputStreamReader(InputStream in) 使用默认字符编码创建InputStreamReader对象
    InputStreamReader(InputStream in,String chatset) 使用指定的字符编码创建InputStreamReader对象
    OutputStreamWriter(OutputStream out) 使用默认字符编码创建OutputStreamWriter对象
    OutputStreamWriter(OutputStream out,String charset) 使用指定的字符编码创建OutputStreamWriter对象
  • 代码演示

    public class ConversionStreamDemo {
        public static void main(String[] args) throws IOException {
      OutputStreamWriter osw = new OutputStreamWriter(new                              FileOutputStream("osw.txt"));
      OutputStreamWriter osw = new OutputStreamWriter(new                          FileOutputStream("myCharStream\\osw.txt"),"GBK");
            osw.write("中国");
            osw.close();
    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();
        }
    }

字符流写数据的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) 写一个字符串的一部分
  • 刷新和关闭的方法

    方法名 说明
    flush() 刷新流,之后还可以继续写数据
    close() 关闭流,释放资源,但是在关闭之前会先刷新流。一旦关闭,就不能再写数据
  • 代码演示

    public class OutputStreamWriterDemo {
        public static void main(String[] args) throws IOException {
            OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("myCharStream\\osw.txt"));
            //void write(int c):写一个字符
            osw.write(97);
            osw.write(98);
            osw.write(99);
            //void writ(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();
        }
    }

1.6字符流读数据的2种方式【应用】

  • 方法介绍

    方法名 说明
    int read() 一次读一个字符数据
    int read(char[] cbuf) 一次读一个字符数组数据
  • 代码演示

    public class InputStreamReaderDemo {
        public static void main(String[] args) throws IOException {
            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();
        }
    }

字符缓冲流【应用】

  • 字符缓冲流(一行)介绍

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

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

  • 构造方法

    方法名 说明
    BufferedWriter(Writer out) 创建字符缓冲输出流对象
    BufferedReader(Reader in) 创建字符缓冲输入流对象
  • 代码演示

  • package io;
    import java.io.*;
    public class Demo2 {
        public static void main(String[] args) throws IOException {
          //文件输入到内存
           Reader reader = new FileReader("d:/wy.txt");
           Writer writer=new FileWriter("d:/wyy.txt");
           BufferedReader br=new BufferedReader(reader);
           BufferedWriter bw=new BufferedWriter(writer);
           StringBuffer sb= new StringBuffer();//用于读取数据的拼接
           String line=null;
           while((line=br.readLine())!=null){
              sb.append(line);
           }
           //在内存中替换占位符
           String str=sb.toString();//将stringBuffered转换为String
           str= str.replace("{name}","wy");
           //输出到文件
           bw.write(str);
           if (bw!=null)bw.close();
           if (br!=null)br.close();
        }
    }
    ​

字符缓冲流特有功能

  • 方法介绍

    BufferedWriter:

    方法名 说明
    void newLine() 写一行行分隔符,行分隔符字符串由系统属性定义

    BufferedReader:

    方法名 说明
    String readLine() 读一行文字。 结果包含行的内容的字符串,不包括任何行终止字符如果流的结尾已经到达,则为null
  • 代码演示

    public class BufferedStreamDemo02 {
        public static void main(String[] args) throws IOException {
    ​
            //创建字符缓冲输出流
            BufferedWriter bw = new BufferedWriter(new                                              FileWriter("myCharStream\\bw.txt"));
            //写数据
            for (int i = 0; i < 10; i++) {
                bw.write("hello" + i);
                //bw.write("\r\n");
                bw.newLine();
                bw.flush();
            }
            //释放资源
            bw.close();
            //创建字符缓冲输入流
            BufferedReader br = new BufferedReader(new                                              FileReader("myCharStream\\bw.txt"));
            String line;
            while ((line=br.readLine())!=null) {
                System.out.println(line);
            }
            br.close();
        }
    }

7、特殊流

标准输入流

System类中有两个静态的成员变量

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

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

public class SystemInDemo {
    public static void main(String[] args) throws IOException {
        //public static final InputStream in:标准输入流
        InputStream is = System.in;
        int by;
        while ((by=is.read())!=-1) {
            System.out.print((char)by);
        }
​
        //如何把字节流转换为字符流?用转换流
        InputStreamReader isr = new InputStreamReader(is);
        //使用字符流能不能够实现一次读取一行数据呢?可以
        //但是,一次读取一行数据的方法是字符缓冲输入流的特有方法
        BufferedReader br = new BufferedReader(isr);
        BufferedReader br = new BufferedReader(new                                                                  InputStreamReader(System.in));
        System.out.println("请输入一个字符串:");
        String line = br.readLine();
        System.out.println("你输入的字符串是:" + line);
​
        System.out.println("请输入一个整数:");
        int i = Integer.parseInt(br.readLine());
        System.out.println("你输入的整数是:" + i);
​
        //自己实现键盘录入数据太麻烦了,所以Java就提供了一个类供我们使用
        Scanner sc = new Scanner(System.in);
    }}

 

标准输出流

  • System类中有两个静态的成员变量

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

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

  • 输出语句的本质:是一个标准的输出流

    • PrintStream ps = System.out;

    • PrintStream类有的方法,System.out都可以使用

public class SystemOutDemo {
    public static void main(String[] args) {
        //public static final PrintStream out:标准输出流
        PrintStream ps = System.out;
​
        //能够方便地打印各种数据值
        ps.print("hello");
        ps.print(100);
​
        ps.println("hello");
        ps.println(100);
​
        //System.out的本质是一个字节输出流
        System.out.println("hello");
        System.out.println(100);
​
        System.out.println();
        System.out.print();
    }
}

字节打印流

  • 打印流分类

    • 字节打印流:PrintStream

    • 字符打印流:PrintWriter

  • 打印流的特点

    • 只负责输出数据,不负责读取数据

    • 永远不会抛出IOException

    • 有自己的特有方法

  • 字节打印流

    • PrintStream(String fileName):使用指定的文件名创建新的打印流

    • 使用继承父类的方法写数据,查看的时候会转码;使用自己的特有方法写数据,查看的数据原样输出

    • 可以改变输出语句的目的地

      public static void setOut(PrintStream out):重新分配“标准”输出流

  • 示例代码

    public class SystemOutDemo {
        public static void main(String[] args) {
            //public static final PrintStream out:标准输出流
            PrintStream ps = System.out;
    ​
            //能够方便地打印各种数据值
            ps.print("hello");
            ps.print(100);
    ​
            ps.println("hello");
            ps.println(100);
    ​
            //System.out的本质是一个字节输出流
            System.out.println("hello");
            System.out.println(100);
    ​
            System.out.println();
            System.out.print();
        }
    }

字符打印流

  • 字符打印流构造房方法

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

    public class PrintWriterDemo {
        public static void main(String[] args) throws IOException {
            //PrintWriter(String fileName) :使用指定的文件名创建一个新的         PrintWriter,而不需要自动执行行刷新
            PrintWriter pw = new PrintWriter("myOtherStream\\pw.txt");
    ​
            pw.write("hello");
            pw.write("\r\n");
            pw.flush();
            pw.write("world");
            pw.write("\r\n");
            pw.flush();
    ​
            pw.println("hello");
            /*
                pw.write("hello");
                pw.write("\r\n");
             */
    //        pw.flush();
    //        pw.println("world");
    //        pw.flush();
    ​
            //PrintWriter(Writer out, boolean autoFlush):创建一个新的PrintWriter
            PrintWriter pw = new PrintWriter(new FileWriter("myOtherStream\\pw.txt"),true);
    //        PrintWriter pw = new PrintWriter(new FileWriter("myOtherStream\\pw.txt"),false);
    ​
            pw.println("hello");
            /*
                pw.write("hello");
                pw.write("\r\n");
                pw.flush();
             */
            pw.println("world");
    ​
            pw.close();
        }
    }

对象序列化流

  • 对象序列化介绍

    • 对象序列化:就是将对象保存到磁盘中,或者在网络中传输对象

    • 这种机制就是使用一个字节序列表示一个对象,该字节序列包含:对象的类型、对象的数据和对象中存储的属性等信息

    • 字节序列写到文件之后,相当于文件中持久保存了一个对象的信息

    • 反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化

  • 对象序列化流: ObjectOutputStream

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

  • 构造方法

    方法名 说明
    ObjectOutputStream(OutputStream out) 创建一个写入指定的OutputStream的ObjectOutputStream
  • 序列化对象的方法

    方法名 说明
    void writeObject(Object obj) 将指定的对象写入ObjectOutputStream

注意事项

  • 一个对象要想被序列化,该对象所属的类必须必须实现Serializable 接口

  • Serializable是一个标记接口,实现该接口,不需要重写任何方法

对象反序列化流

  • 对象反序列化流: ObjectInputStream

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

  • 构造方法

    方法名 说明
    ObjectInputStream(InputStream in) 创建从指定的InputStream读取的ObjectInputStream
  • 反序列化对象的方法

    方法名 说明
    Object readObject() 从ObjectInputStream读取一个对象

serialVersionUID&transient

  • serialVersionUID

    • 用对象序列化流序列化了一个对象后,假如我们修改了对象所属的类文件,读取数据会不会出问题呢?

      • 会出问题,会抛出InvalidClassException异常

    • 如果出问题了,如何解决呢?

      • 重新序列化

      • 给对象所属的类加一个serialVersionUID

        • private static final long serialVersionUID = 42L;

  • transient

    • 如果一个对象中的某个成员变量的值不想被序列化,又该如何实现呢?

      • 给该成员变量加transient关键字修饰,该关键字标记的成员变量不参与序列化过程

Properties集合

Properties作为Map集合的使用

  • Properties介绍

    • 是一个Map体系的集合类

    • Properties可以保存到流中或从流中加载

    • 属性列表中的每个键及其对应的值都是一个字符串

  • Properties基本使用

    public class PropertiesDemo01 {
        public static void main(String[] args) {
            //创建集合对象
            Properties prop = new Properties();
    ​
            //存储元素
            prop.put("itheima001", "林青霞");
            prop.put("itheima002", "张曼玉");
            prop.put("itheima003", "王祖贤");
    ​
            //遍历集合
            Set<Object> keySet = prop.keySet();
            for (Object key : keySet) {
                Object value = prop.get(key);
                System.out.println(key + "," + value);
            }
        }
    }

Properties作为Map集合的特有方法

  • 特有方法

    方法名 说明
    Object setProperty(String key, String value) 设置集合的键和值,都是String类型,底层调用 Hashtable方法 put
    String getProperty(String key) 使用此属性列表中指定的键搜索属性
    Set<String> stringPropertyNames() 从该属性列表中返回一个不可修改的键集,其中键及其对应的值是字符串
  • 示例代码

    public class PropertiesDemo02 {
        public static void main(String[] args) {
            //创建集合对象
            Properties prop = new Properties();
    ​
            //Object setProperty(String key, String value):设置集合的键和值,都是String类型,底层调用Hashtable方法put
            prop.setProperty("itheima001", "林青霞");
            /*
                Object setProperty(String key, String value) {
                    return put(key, value);
                }
    ​
                Object put(Object key, Object value) {
                    return map.put(key, value);
                }
             */
            prop.setProperty("itheima002", "张曼玉");
            prop.setProperty("itheima003", "王祖贤");
    ​
            //String getProperty(String key):使用此属性列表中指定的键搜索属性
    //        System.out.println(prop.getProperty("itheima001"));
    //        System.out.println(prop.getProperty("itheima0011"));
    ​
    //        System.out.println(prop);
    ​
            //Set<String> stringPropertyNames():从该属性列表中返回一个不可修改的键集,其中键及其对应的值是字符串
            Set<String> names = prop.stringPropertyNames();
            for (String key : names) {
    //            System.out.println(key);
                String value = prop.getProperty(key);
                System.out.println(key + "," + value);
            }
        }
    }

Properties和IO流相结合的方法

  • 和IO流结合的方法

    方法名 说明
    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)方法的格式写入输出字符流
  • 示例代码

    public class PropertiesDemo03 {
        public static void main(String[] args) throws IOException {
            //把集合中的数据保存到文件
    //        myStore();
    ​
            //把文件中的数据加载到集合
            myLoad();
    ​
        }
    ​
        private static void myLoad() throws IOException {
                Properties prop = new Properties();
    ​
            //void load(Reader reader):
            FileReader fr = new FileReader("myOtherStream\\fw.txt");
            prop.load(fr);
            fr.close();
    ​
            System.out.println(prop);
        }
    ​
        private static void myStore() throws IOException {
            Properties prop = new Properties();
    ​
            prop.setProperty("itheima001","林青霞");
            prop.setProperty("itheima002","张曼玉");
            prop.setProperty("itheima003","王祖贤");
    ​
            //void store(Writer writer, String comments):
            FileWriter fw = new FileWriter("myOtherStream\\fw.txt");
            prop.store(fw,null);
            fw.close();
        }
    }

本文含有隐藏内容,请 开通VIP 后查看

网站公告

今日签到

点亮在社区的每一天
去签到