IO流知识总结

File类

File类概述和构造方法

  • File类介绍
    • 它是文件和目录路径名的抽象表示
    • 文件和目录是可以通过File封装成对象的
    • 对于File而言,其封装的并不是一个真正的文件,仅仅是一个路劲名而已,它可以是存在的也可以是不存在的,将来是要通过具体的操作方法把这个路径的内容转换为具体存在的。
  • File类的构造方法
    在这里插入图片描述
    示例
public class FileDemo01 {
    public static void main(String[] args) {
        //File(String pathname): 通过将给定的路径名字符串转换为抽象路径名来创建新的 File实例
        File f1 = new File("E:\\itcast\\java.txt");
        System.out.println(f1);
 
        //File(String parent, String child): 从父路径名字符串和子路径名字符串创建新的 File实例
        File f2 = new File("E:\\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);
    }
}

绝对路径和相对路径

  • 绝对路径
    是一个完整的路径,从盘符开始
  • 相对路劲
    是一个简化的路径,相对当前项目下的路径
    示例
public class FileDemo02 {
    public static void main(String[] args) {
        // 是一个完整的路径,从盘符开始
        File file1 = new File("D:\\itheima\\a.txt");
 
        // 是一个简化的路径,从当前项目根目录开始
        File file2 = new File("a.txt");
        File file3 = new File("模块名\\a.txt");
    }
}

File类创建功能

在这里插入图片描述
mkdir()创建单极文件夹
1.只能创建单极,不能创建多级文件夹
2.不管调用者有没有后缀名,只能创建单级文件夹
mkdirs()创建多级文件夹
1.可以创建单级也可创建多级文件夹
2.只能创建文件夹。
一般只用mkdirs()
createNewFile()
1.它只能创建文件,所以必须在存在的文件夹下执行
2.如果文件夹不存在,那么需要先创建一个文件夹才行。

File file = new File("filemodule\\aa");
if(!file.exists()){
file.mkdirs();
}
File newFile = new File(file,"a.txt");
newFile.createNewFile();
ublic class FileDemo02 {
    public static void main(String[] args) throws IOException {
        //需求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.mkdir());
        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.mkdir());
        System.out.println(f4.createNewFile());
    }
}

File类删除功能

在这里插入图片描述注意点
1.不走回收站
2.如果删除的是文件,那么直接删除,如果删除的是文件夹,那么只能删除空文件夹
3.如果要删除一个有内容的文件夹,只能先进入这个文件夹,把里边的内容全部删除完毕,才能再次删除这个文件夹
简单来说,只能删除文件和空文件夹。

public class FileDemo03 {
    public static void main(String[] args) throws IOException {
//        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文件
        System.out.println(f1.delete());
        System.out.println("‐‐‐‐‐‐‐‐");
 
        //需求3:在当前模块目录下创建itcast目录
        File f2 = new File("myFile\\itcast");
//        System.out.println(f2.mkdir());
 
        //需求4:删除当前模块目录下的itcast目录
        System.out.println(f2.delete());
        System.out.println("‐‐‐‐‐‐‐‐");
 
        //需求5:在当前模块下创建一个目录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());
 
        //需求6:删除当前模块下的目录itcast
        System.out.println(f4.delete());
        System.out.println(f3.delete());
    }
}

File判断功能和获取功能

判断功能
在这里插入图片描述获取功能
在这里插入图片描述
getName()
1.如果调用者是文件夹,那么获取的是文件夹和后缀名
2.如果调用者是一个文件夹,那么获取的是文件夹的名字
listFiles()
进入文件夹,获取这个文件夹里边所有的文件和文件夹的File对象,并把这些File对象都放在一个数组中返回。包括隐藏文件夹和文件都可以获取。
注意:
1.当调用者是一个文件时,会返回null
2.当调用者是一个空文件夹时,返回一个长度为0的数组
3.当调用者是一个有内容的文件夹时,会获取所有文件放入一个数组中
4.当调用者是一个有访问权限的文件夹时,会返回null

public class FileDemo04 {
    public static void main(String[] args) {
        //创建一个File对象
        File f = new File("myFile\\java.txt");
 
//        public boolean isDirectory():测试此抽象路径名表示的File是否为目录
//        public boolean isFile():测试此抽象路径名表示的File是否为文件
//        public boolean exists():测试此抽象路径名表示的File是否存在
        System.out.println(f.isDirectory());
        System.out.println(f.isFile());
        System.out.println(f.exists());
 
//        public String getAbsolutePath():返回此抽象路径名的绝对路径名字符串
//        public String getPath():将此抽象路径名转换为路径名字符串
//        public String getName():返回由此抽象路径名表示的文件或目录的名称
        System.out.println(f.getAbsolutePath());
        System.out.println(f.getPath());
        System.out.println(f.getName());
        System.out.println("‐‐‐‐‐‐‐‐");
 
//        public File[] listFiles():返回此抽象路径名表示的目录中的文件和目录的File对象数组
        File f2 = new File("E:\\itcast");
        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对象
    • 遍历这个File对象,获取它下边的每个文件和文件夹对象
    • 判断当前遍历的File对象是文件还是文件夹
    • 如果是文件,直接删除
    • 如果是文件夹,递归调用自己,将当前遍历到的File对象作为参数传递
    • 参数传递过来的文件夹File对象已经处理完成,最后直接删除这个空文件夹
public class Test2{
public static void main(String[] args){
File src = new File("C:\\Users\\apple\\Desktop\\src");
deleteDir(src);
}
//1.创建一个方法,接收一个File对象
private static void deleDir(File src){
//先删除这个文件夹里边所有的内容
//递归 方法在方法体中自己调用自己
//注意:可以解决所有文件夹和递归相结合的题目
//2.遍历这个FIle对象,获取它下边的每个文件和文件夹对象
File[] files = src.listFiles();
//3.判断当前遍历到的FIle对象是文件还是文件夹
for(File file : flies){
if(file.isFile()){
//4.如果是文件,直接删除
file.delete();
}else{
//5.如果是文件夹,递归调用自己,将当前遍历到的FIle对象当作参数传递
deleteDir(file);//参数一定要是src文件夹里的文件夹File对象
}
}
//6.参数传递过来的文件夹FIle对象已经处理完成,最后直接删除这个空文件夹
sre.delete();
}
}

统计文件夹中每种文件的个数

ublic class Test3 {
    public static void main(String[] args) {
        //统计一个文件夹中,每种文件出现的次数.
        //统计 ‐‐‐ 定义一个变量用来统计. ‐‐‐‐ 弊端:同时只能统计一种文件
        //利用map集合进行数据统计,键 ‐‐‐ 文件后缀名  值 ‐‐‐‐  次数
 
        File file = new File("filemodule");
        HashMap<String, Integer> hm = new HashMap<>();
        getCount(hm, file);
        System.out.println(hm);
    }
  
//1.定义一个方法,参数是HashMap集合用来统计次数和File对象要统计的文件夹     
    private static void getCount(HashMap<String, Integer> hm, File file) {
       //2.遍历File对象,获取它下边的每一个文件和文件夹对象   
        File[] files = file.listFiles();
        for (File f : files) {
           //3.判断当前File对象是文件还是文件夹   
            if(f.isFile()){
               //如果是文件,判断这种类型文件后缀名在HashMap集合中是否出现过   
                String fileName = f.getName();
                String[] fileNameArr = fileName.split("\\.");
                if(fileNameArr.length == 2){
                    String fileEndName = fileNameArr[1];
                    if(hm.containsKey(fileEndName)){
                        //出现过,获取这种类型文件的后缀名出现的次数,对其+1,在存回集合中
                        Integer count = hm.get(fileEndName);
                        //这种文件又出现了一次.
                        count++;
                        //把已经出现的次数给覆盖掉.
                        hm.put(fileEndName,count);
                    }else{
                        // 没出现过,将这种类型文件的后缀名存入集合中,次数存1
                        hm.put(fileEndName,1);
                    }
                }
            }else{
              //如果是文件夹,递归调用自己,HashMap集合就是参数集合,File对象是当前文件夹对象代码实getCount(hm,f);
            }
        }
    }

常用方法

public String getAbsolutePath() :返回此File的绝对路径名字符串。

public String getPath() :将此File转换为路径名字符串。

public String getName() :返回由此File表示的文件或目录的名称。

public long length() :返回由此File表示的文件的长度。

public boolean exists() :此File表示的文件或目录是否实际存在。

public boolean isDirectory() :此File表示的是否为目录。

public boolean isFile() :此File表示的是否为文件。

public boolean createNewFile() :当且仅当具有该名称的文件尚不存在时,创建一个新的空文件。

public boolean delete() :删除由此File表示的文件或目录。

public boolean mkdir() :创建由此File表示的目录。

public boolean mkdirs() :创建由此File表示的目录,包括任何必需但不存在的父目录。

IO流概述和分类

  • IO流介绍
    • IO:输入输出
    • 流:是一种抽象概念,是对数据传输的总称,也就是说数据在设备间的传输称为流,流的本质是数据传输
    • IO流就是用来处理设备间数据传输问题的,常见应用:文件复制;文件上传,文件下载
  • IO流的分类
    • 按照数据的流向
      • 输入流:读数据
      • 输出流:写数据
  • 按照数据类型来分
    • 字节流
      • 字节输入流
      • 字节输出流
    • 字符流
      • 字符输入流
      • 字符输出流
  • IO流的使用场景
    • 如果操作纯文本文件,优先使用字符流
    • 如果操作图片、视频、音频等二进制文件,优先使用字节流
    • 如果不确定文件类型,优先使用字节流,字节流是万能的流

字节流写数据

  • 字节流抽象基类
    • InputStream:这个抽象类是表示字节输入流的所有类的超类
    • OutputStream:这个抽象类是表示字节输出流的所有类的超类
    • 子类名特点:子类名称都是以其父类名作为子类名的后缀
  • 字节输出流
    • FileOutStream(String name):创建文件输出流以指定的名称写入文件
  • 使用字节输出流写数据的步骤
    • 创建字节输出流对象(调用系统功能创建了文件,创建字节输出流对象,让字节输出流指向文件)
    • 调用字节输出流对象的写数据方法
    • 释放资源(关闭此文件输出流并释放与此流相关联的任何系统资源)
      示例代码
public class FileOutputStreamDemo01 {
    public static void main(String[] args) throws IOException {
    //创建字节输出流对象
       /*   
       注意点:       
       1.如果文件不存在,会帮我们创建               
       2.如果文件存在,会把文件清空               
       */   
       //FileOutputStream(String name):创建文件输出流以指定的名称写入文件   
        FileOutputStream fos = new FileOutputStream("myByteStream\\fos.txt");
 
        //void write(int b):将指定的字节写入此文件输出流
        fos.write(97);
//        fos.write(57);
//        fos.write(55);
 
        //最后都要释放资源
        //void close():关闭此文件输出流并释放与此流相关联的任何系统资源。
        fos.close();
    }
}
    

字节流写数据的三种方式

  • 写数据的方法分类
    在这里插入图片描述
public class FileOutputStreamDemo02 {
    public static void main(String[] args) throws IOException {
        //FileOutputStream(String name):创建文件输出流以指定的名称写入文件
        FileOutputStream fos = new FileOutputStream("myByteStream\\fos.txt");
        //FileOutputStream(File file):创建文件输出流以写入由指定的 File对象表示的文件
//        FileOutputStream fos = new FileOutputStream(new File("myByteStream\\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[] 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();
    }
}

字节流写数据的两个问题

  • 字节流写数据如何实现换行
    • windows:\r\n
    • linux:\n
    • mac:\r
  • 字节流写数据如何实现追加写入
    • public FileOutputStream(String name,boolean append)
    • 创建文件输出流以指定名称写入文件。如果第二个参数为true,则字节将写入文件的末尾而不是开头
ublic class FileOutputStreamDemo03 {
    public static void main(String[] args) throws IOException {
        //创建字节输出流对象
//        FileOutputStream fos = new FileOutputStream("myByteStream\\fos.txt");
        FileOutputStream fos = new FileOutputStream("myByteStream\\fos.txt",true);
 
        //写数据
        for (int i = 0; i < 10; i++) {
            fos.write("hello".getBytes());
            fos.write("\r\n".getBytes());
        }
 
        //释放资源
        fos.close();
    }
}

字节流写数据加异常处理

  • 异常处理格式
    • try-catch-finally
try{
可能出现异常的代码;     
}catch(异常类名 变量名){
异常的处理代码;     
}finally{
执行所有清除操作;     
}
  • finally特点
    • 被finally控制的语句一定会执行,除非JVM推出
public class FileOutputStreamDemo04 {
    public static void main(String[] args) {
        //加入finally来实现释放资源
        FileOutputStream fos = null;
        try {
            fos = new FileOutputStream("myByteStream\\fos.txt");
            fos.write("hello".getBytes());
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if(fos != null) {
                try {
                    fos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

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

  • 字节输入流
    • FileInputStream(String):通过打开与实际文件的连接来创建一个FileInputStream,该文件由文件系统中的路径名name命名
  • 字节输入流读取数据的步骤
    • 创建字节输入流对象
    • 调用字节输入流对象的读数据方法
    • 释放资源
public class FileInputStreamDemo01 {
    public static void main(String[] args) throws IOException {
        //创建字节输入流对象
        //FileInputStream(String name)
        FileInputStream fis = new FileInputStream("myByteStream\\fos.txt");
 
        int by;
        /*
            fis.read():读数据
            by=fis.read():把读取到的数据赋值给by
            by != ‐1:判断读取到的数据是否是‐1
         */
        while ((by=fis.read())!=1) {
            System.out.print((char)by);
        }
 
        //释放资源
        fis.close();
    }
}

字节流复制文件

 
        FileInputStream fis = new FileInputStream("myByteStream\\fos.txt");
 
        int by;
        /*
            fis.read():读数据
            by=fis.read():把读取到的数据赋值给by
            by != ‐1:判断读取到的数据是否是‐1
         */
        while ((by=fis.read())!=1) {
            System.out.print((char)by);
        }
 
        //释放资源
        fis.close();
    }
}
public class CopyTxtDemo {
    public static void main(String[] args) throws IOException {
        //根据数据源创建字节输入流对象
        FileInputStream fis = new FileInputStream("E:\\itcast\\窗里窗外.txt");
        //根据目的地创建字节输出流对象
        FileOutputStream fos = new FileOutputStream("myByteStream\\窗里窗外.txt");
 
        //读写数据,复制文本文件(一次读取一个字节,一次写入一个字节)
        int by;
        while ((by=fis.read())!=1) {
            fos.write(by);
        }
 
        //释放资源
        fos.close();
        fis.close();
    }
}

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

  • 一次读一个字节数组的方法
    • public int read(byte[] b): 从输入流读取最多b.length个字节的数据
    • 返回的是读入缓冲区的总字节数,也就是实际的读取字节个数
public class FileInputStreamDemo02 {
    public static void main(String[] args) throws IOException {
        //创建字节输入流对象
        FileInputStream fis = new FileInputStream("myByteStream\\fos.txt");
 
        byte[] bys = new byte[1024]; //1024及其整数倍
        int len;
       //循环读取   
        while ((len=fis.read(bys))!=1) {
            System.out.print(new String(bys,0,len));
        }
 
        //释放资源
        fis.close();
    }
}

字节缓冲流

字节缓冲流构造方法

  • 字节缓冲流介绍
    • BufferedOutStream:该类实现缓冲输出流,通过设置这样的输出流,应用程序可以向底层输出流写入字节,而不必为写入的每个字节导致底层系统的调用
    • BufferredInputStream: 创建BufferedInputStream将创建一个内部缓冲区数组.当从流中读取或跳过字节时,内部缓冲区将根据需要从所包含的输入流中重新填充,一次很多字节
      在这里插入图片描述
public class BufferStreamDemo {
    public static void main(String[] args) throws IOException {
        //字节缓冲输出流:BufferedOutputStream(OutputStream out)
 
        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();
    }
}
        

在这里插入图片描述

字符流

为什么出现字符流

  • 字符流的介绍
    由于字节流操作中文不是特别的方便,所以Java就提供字符流
    字符流 = 字节流 + 编码表
  • 中文的字节存储方式
    用字节流复制文本文件时,文本文件也会有中文,但是没有问题,原因是 最终底层操作会自动进行字节拼接成中文,如何识别是中文的呢?
    汉字在存储的时候,无论选择哪种编码存储,第一个字节都是负数

编码表

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

常见字符集

	- ASCII字符集
	- GBXXX字符集
	- Unicode字符集
	- 编码规则:
		128个Us-ASCII字符,只需要一个字节编码
		拉丁文等字符,需要两个字节编码
		大部分常用字,使用三个字节编码
		其他极少使用Unicode辅助字符,使用四字节编码		

windows默认使用码表为:GBK,一个字符两个字节
idea和工作中默认UTF-8,一个中文三个字节。

字符串中的编码解码问题

  • 相关方法
    在这里插入图片描述
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);
    }
}

字符流写数据

  • 介绍
    Writer:用于写入字符流的抽象父类
    FileWriter:用于写入字符流的常用子类
  • 构造方法
    在这里插入图片描述
    如果文件不存在,那么会创建,但是父级路径要存在
    写数据:
    1.写出int类型的整数,实际写出的是整数在码表上对应的字母。
    2.写出字符串数据,是把字符串本身原样写出
    3.每次写完必须要释放资源
  • 成员方法
    在这里插入图片描述
  • 刷新和关闭方法
    在这里插入图片描述
    刷新流就是将数据刷新到本地文件。
public class OutputStreamWriterDemo {
    public static void main(String[] args) throws IOException {
        FileWriter fw = new FileWriter("myCharStream\\a.txt");
 
        //void write(int c):写一个字符
//        fw.write(97);
//        fw.write(98);
//        fw.write(99);
 
        //void writ(char[] cbuf):写入一个字符数组
        char[] chs = {'a', 'b', 'c', 'd', 'e'};
//        fw.write(chs);
 
        //void write(char[] cbuf, int off, int len):写入字符数组的一部分
//        fw.write(chs, 0, chs.length);
//        fw.write(chs, 1, 3);
 
        //void write(String str):写一个字符串
//        fw.write("abcde");
 
        //void write(String str, int off, int len):写一个字符串的一部分
//        fw.write("abcde", 0, "abcde".length());
        fw.write("abcde", 1, 3);
 
        //释放资源
        fw.close();
    }
}

字符流读数据

  • 介绍
    Reader:用于读取字符流的抽象父类
    FileReader:用于读取字符流的常用子类
  • 构造方法
    在这里插入图片描述
  • 成员方法
    在这里插入图片描述
public class InputStreamReaderDemo {
    public static void main(String[] args) throws IOException {
   
        FileReader fr = new FileReader("myCharStream\\b.txt");
 
        //int read():一次读一个字符数据
//        int ch;
//        while ((ch=fr.read())!=‐1) {
//            System.out.print((char)ch);
//        }
 
        //int read(char[] cbuf):一次读一个字符数组数据
        char[] chs = new char[1024];
        int len;
        while ((len = fr.read(chs)) !=1) {
            System.out.print(new String(chs, 0, len));
        }
 
        //释放资源
        fr.close();
    }
}

字符缓冲流

  • 字符缓冲流介绍
    • BufferedWriter:将文本写入字符输出流,缓冲字符,以提供单个字符,数组和字符串的高效写入,可以指定缓冲区大小,或者可以接受默认大小。默认值足够大,可以用于大多数用途。
    • BufferedReader:从字符输入流读取文本,缓冲字符,以提供字符,数组和行的高效读取,可以直接指定缓冲区大小,或者可以使用默认大小,默认值足够大,可以用于大多数用途。
  • 构造方法
    在这里插入图片描述
public class BufferedStreamDemo01 {
    public static void main(String[] args) throws IOException {
        //BufferedWriter(Writer out)
        BufferedWriter bw = new BufferedWriter(new                                         
                   FileWriter("myCharStream\\bw.txt"));
        bw.write("hello\r\n");
        bw.write("world\r\n");
        bw.close();
 
        //BufferedReader(Reader in)
        BufferedReader br = new BufferedReader(new                                         
                  FileReader("myCharStream\\bw.txt"));
 
        //一次读取一个字符数据
//        int ch;
//        while ((ch=br.read())!=‐1) {
//            System.out.print((char)ch);
//        }
 
        //一次读取一个字符数组数据
        char[] chs = new char[1024];
        int len;
        while ((len=br.read(chs))!=1) {
            System.out.print(new String(chs,0,len));
        }
 
        br.close();
    }
}

字符缓冲流特有功能

  • 方法介绍
    在这里插入图片描述
    在这里插入图片描述
    使用字符缓冲流读取文件中数据,排序后再次写到本地文件
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();
    }
}
public class CharStreamDemo14 {
    public static void main(String[] args) throws IOException {
        //需求:读取文件中的数据,排序后再次写到本地文件
        //分析:
        //1.要把文件中的数据读取进来。
        BufferedReader br = new BufferedReader(new FileReader("charstream\\sort.txt"));
        //输出流一定不能写在这里,因为会清空文件中的内容
       //BufferedWriter bw = new BufferedWriter(new FileWriter("charstream\\sort.txt"));
 
        String line = br.readLine();
        System.out.println("读取到的数据为" + line);
        br.close();
 
        //2.按照空格进行切割
        String[] split = line.split(" ");//9 1 2 5 3 10 4 6 7 8
        //3.把字符串类型的数组变成int类型
        int [] arr = new int[split.length];
        //遍历split数组,可以进行类型转换。
        for (int i = 0; i < split.length; i++) {
            String smallStr = split[i];
            //类型转换
            int number = Integer.parseInt(smallStr);
            //把转换后的结果存入到arr中
            arr[i] = number;
        }
        //4.排序
        Arrays.sort(arr);
        System.out.println(Arrays.toString(arr));
 
        //5.把排序之后结果写回到本地 1 2 3 4...
        BufferedWriter bw = new BufferedWriter(new FileWriter("charstream\\sort.txt"));
        //写出
        for (int i = 0; i < arr.length; i++) {
            bw.write(arr[i] + " ");
            bw.flush();
        }
        //释放资源
        bw.close();
 
    }
}

IO流小结

在这里插入图片描述

转换流

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

  • InputStreamReader:是从字节流到字符流的桥梁,父类是Reader,它读取字节,并使用指定的编码将其解码为字符。它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集
  • OutputStreamWriter:是字符流到字节流的桥梁,父类是Writer是从字符流到字节流的桥梁,使用指定的编码将写入的字符编码为字节。它使用的字符集可以由名称指定,或者接受平台的默认字符集。

转换流读写数据

  • 构造方法
    在这里插入图片描述
public class ConversionStreamDemo {
    public static void main(String[] args) throws IOException {
        //OutputStreamWriter osw = new OutputStreamWriter(new                               
              FileOutputStream("myCharStream\\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();
    }
}

对象操作流

对象序列化流

  • 对象序列化介绍
    • 对象序列化:就是将对象保存到磁盘里,或者在网络中传输对象
    • 这种机制就是使用一个字节序列表示一个对象,该字节序列包括:对象类型、对象的数据和对象中存储的属性等信息
    • 字节序列写到文件之后,相当于文件中持久保存了一个对象的信息
    • 繁殖,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化。
  • 对象序列化流:ObjectOutputStream
    • 将对象的原始数据类型和图形写入OutputStream.可以使用ObjectInputStream读取(重构)对象。可以通过使用流的文件来实现对象的持久化存储。如果流是网络套接字流,则可以在另一台主机上或另一个进程中重构对象。
      在这里插入图片描述
      注意
    • 一个对象要想被序列化,该对象所属的类必须实现Serializable接口
    • Serializable是一个标记接口,实现该接口,不需要重写任何方法
        isr.close();
    }
}
public class Student implements Serializable {
    private String name;
    private int age;
 
    public Student() {
    }
 
    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }
 ........
public class ObjectOutputStreamDemo {
    public static void main(String[] args) throws IOException {
        //ObjectOutputStream(OutputStream out):创建一个写入指定的OutputStream的
ObjectOutputStream
        ObjectOutputStream oos = new ObjectOutputStream(new 
FileOutputStream("myOtherStream\\oos.txt"));
 
        //创建对象
        Student s = new Student("佟丽娅",30);
 
        //void writeObject(Object obj):将指定的对象写入ObjectOutputStream
        oos.writeObject(s);
 
        //释放资源
        oos.close();
    }
}

对象反序列化流

  • 对象反序列化流:ObjectInputStream
    • ObjectInputStream反序列化前先使用ObjectOutputStream编写原始数据对象
  • 构造方法
    在这里插入图片描述
  • 反序列化对象方法
    在这里插入图片描述
ublic class ObjectInputStreamDemo {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        //ObjectInputStream(InputStream in):创建从指定的InputStream读取的ObjectInputStream
        ObjectInputStream ois = new ObjectInputStream(new 
FileInputStream("myOtherStream\\oos.txt"));
 
        //Object readObject():从ObjectInputStream读取一个对象
        Object obj = ois.readObject();
 
        Student s = (Student) obj;
        System.out.println(s.getName() + "," + s.getAge());
 
        ois.close();
    }
}

serialVersionUID&transient

  • serialVersionUID
    • 用对象序列化流序列化一个对象之后,加入我们修改了对象所属的类文件,读取会出问题,抛出InvalidClassException异常
    • 如何解决
      • 重新序列化
      • 给对象所属的类加一个serialVersionUID
        • private static final long serialVersionUID = 42L;
  • transient
    • 如果一个对象中的某个成员变量的值不想被序列化,该如何
      • 给该成员变量加transient关键字修饰,该关键字标记的成员变量不参与序列化过程。

对象操作流练习

  • 创建多个学生类对象写到文件中,再次读取到内存中
    创建序列化流对象
    创建多个学生对象
    将学生对象添加到集合中
    将集合对象序列化到文件中
    创建反序列化流对象
    将文件中的对象数据,读取到内存中

学生类

public class Student implements Serializable{
    
    private static final long serialVersionUID = 2L;
 
    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;
    }
}

测试类

public class Demo03 {
    /**
     *  read():
     *      读取到文件末尾返回值是 ‐1
     *  readLine():
     *      读取到文件的末尾返回值 null
     *  readObject():
     *      读取到文件的末尾 直接抛出异常
     *  如果要序列化的对象有多个,不建议直接将多个对象序列化到文件中,因为反序列化时容易出异常
     *      建议: 将要序列化的多个对象存储到集合中,然后将集合序列化到文件中
     */
    public static void main(String[] args) throws Exception {
        /*// 序列化
        //1.创建序列化流对象
        ObjectOutputStream oos = new ObjectOutputStream(new 
FileOutputStream("myCode\\oos.txt"));
        ArrayList<Student> arrayList = new ArrayList<>();
        //2.创建多个学生对象
        Student s = new Student("佟丽娅",30);
        Student s01 = new Student("佟丽娅",30);
        //3.将学生对象添加到集合中
        arrayList.add(s);
        arrayList.add(s01);
        //4.将集合对象序列化到文件中
        oos.writeObject(arrayList);
        oos.close();*/
 
        // 反序列化
       //5.创建反序列化流对象   
        ObjectInputStream ois = new ObjectInputStream(new 
FileInputStream("myCode\\oos.txt"));
       //6.将文件中的对象数据,读取到内存中   
        Object obj = ois.readObject();
        ArrayList<Student> arrayList = (ArrayList<Student>)obj;
        ois.close();
        for (Student s : arrayList) {
            System.out.println(s.getName() + "," + s.getAge());
        }
    }
}

Properties作为Map集合的使用

  • Properties介绍
    • 是一个Map体系的集合类
    • Properties可以保存到流中或从流中加载
    • 属性列表中的每个键及其对应的值都是一个字符串
  • Properties基本使用
public class PropertiesDemo01 {
    public static void main(String[] args) {
        //创建集合对象
//        Properties<String,String> prop = new Properties<String,String>(); //错误
        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集合的特有方法

properties创建时是不写泛型的,一般只存储字符串。
properties读取的文件以properties为后缀。里边写键=值,不加其他符号。

  • 特有方法
    在这里插入图片描述
 
public class PropertiesDemo01 {
    public static void main(String[] args) {
        //创建集合对象
//        Properties<String,String> prop = new Properties<String,String>(); //错误
        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);
        }
    }
}
public class PropertiesDemo02 {
    public static void main(String[] args) {
//创建集合对象
        Properties prop = new Properties();
 
        //Object setProperty(String key, String value):设置集合的键和值,都是String类型
        prop.setProperty("itheima001", "佟丽娅");
        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流相结合的方法
    在这里插入图片描述
    String comments是注释字段
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();
    }
}
        
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值