day21(IO(字符流) & FileReader & FileInputStream & InputStreamReader & BufferedReader笔记

21.01_IO流(字符流FileReader)

  • 1.字符流是什么

    • 字符流是可以直接读写字符的IO流
    • 字符流读取字符, 就要先读取到字节数据, 然后转为字符. 如果要写出字符, 需要把字符转为字节写出.
  • 2.FileReader

    • FileReader类的read()方法可以按照字符大小读取

            /**
         * @param args
         * @throws IOException 
         */
        public static void main(String[] args) throws IOException {
            //demo1();
            FileReader fr = new FileReader("xxx.txt");
            int c;
            
            while((c = fr.read()) != -1) {        //通过项目默认的码表一次读取一个字符,gbk码表发现第一个字节是负数(说明是中文),就一次读两个字节,即一个字符
                System.out.print((char)c);        //强转,大家好
            }
            
            fr.close();
        }
      
        public static void demo1() throws FileNotFoundException, IOException {
            FileReader fr = new FileReader("xxx.txt");
            int x = fr.read();    //读取一个字符,22823
            System.out.println(x);
      
            char c = (char)x;    //向下强转,大(大家好)
            System.out.println(c);
            fr.close();
        }
      

21.02_IO流(字符流FileWriter)

  • FileWriter类的write()方法可以自动把字符转为字节写出

      /**
       * @param args
       * @throws IOException 
       */
      public static void main(String[] args) throws IOException {
          FileWriter fw = new FileWriter("yyy.txt");    //创建或清空,若想不清空,写第二个参数true
          fw.write("大家好,基础班快接近尾声了,大家要努力,要坚持!!!");        //写的是一堆字节,电脑用编码表翻译了
          fw.write(97);    //a
          fw.close();
          }
      }
    

21.03_IO流(字符流的拷贝)

public static void demo1() throws FileNotFoundException, IOException {
    FileReader fr = new FileReader("xxx.txt");
    FileWriter fw = new FileWriter("zzz.txt");
    
    int c;
    while((c = fr.read()) != -1) {        //此处不需要char进行强转,因为会自动补16个零
        fw.write(c);
    }
    
    fr.close();
    fw.close();        //Writer类中有一个2k的小缓冲区,如果不关流,就会将内容写到缓冲区里,关流会将缓冲区内容刷新,再关闭
}

21.04_IO流(什么情况下使用字符流)

  • 字符流(FileReader等)也可以拷贝文本文件, 但不推荐使用. 因为读取时会把字节转为字符, 写出时还要把字符转回字节.

  • 字节流拷贝文件(FileInputStream等)直接传输字节

  • 程序需要读取一段文本, 或者需要写出一段文本的时候可以使用字符流 (只读或只写)

  • 读取的时候是按照字符的大小读取的,不会出现半个中文

  • 写出的时候可以直接将字符串写出,不用转换为字节数组

21.05_IO流(字符流是否可以拷贝非纯文本的文件)

  • 不可以拷贝非纯文本的文件

  • 因为在读的时候会将字节转换为字符,在转换过程中,可能找不到对应的字符,就会用?代替,写出的时候会将字符转换成字节写出去

  • 如果是?,直接写出,这样写出之后的文件就乱了,看不了了

21.06_IO流(自定义字符数组的拷贝)

    FileReader fr = new FileReader("aaa.txt");            //创建字符输入流,关联aaa.txt
    FileWriter fw = new FileWriter("bbb.txt");            //创建字符输出流,关联bbb.txt
    
    int len;
    char[] arr = new char[1024*8];    //自定义字符数组的拷贝                    //创建字符数组
    while((len = fr.read(arr)) != -1) {                    //将数据读到字符数组中
        fw.write(arr, 0, len);                            //从字符数组将数据写到文件上
    }
    
    fr.close();        //关流释放资源
    fw.close();        //不知道为什么,没法写进,文件打开会卡死

21.07_IO流(带缓冲的字符流)

BufferedReader的read()方法读取字符时会一次读取若干字符到缓冲区, 然后逐个返回给程序, 降低读取文件的次数, 提高效率

BufferedWriter的write()方法写出字符时会先写到缓冲区, 缓冲区写满时才会写到文件, 降低写文件的次数, 提高效率

  •   BufferedReader br = new BufferedReader(new FileReader("aaa.txt"));    //创建字符输入流对象,关联aaa.txt,缓冲区8192(int值),16k
      BufferedWriter bw = new BufferedWriter(new FileWriter("bbb.txt"));    //创建字符输出流对象,关联bbb.txt
      
      int ch;                
      while((ch = br.read()) != -1) {        //read一次,会先将缓冲区读满,从缓冲去中一个一个的返给临时变量ch
          bw.write(ch);        //write一次,是将数据装到字符数组,装满后再一起写出去
      }
      
      br.close();            //关流
      bw.close();  
    

21.08_IO流(readLine()和newLine()方法)

  • BufferedReader的readLine()方法可以读取一行字符(不包含换行符号)

    • 读取一个文本行。通过下列字符之一即可认为某行已终止:换行(\n)、回车(\r)或回车后直接跟着换行。
  • BufferedWriter的newLine()可以输出一个跨平台的换行符号"\r\n"

    • 写入一个行分隔符

        /**
         * @param args
         * 带缓冲区的流中的特殊方法
         * readLine()
         * newLine();
         * 
         * newLine()与\r\n的区别
             * newLine()是跨平台的方法
             * \r\n只支持的是windows系统
         * @throws IOException 
         */
        public static void main(String[] args) throws IOException {
            //demo1();
            BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));    //对FileReader包装
            BufferedWriter bw = new BufferedWriter(new FileWriter("aaa.txt"));
            
            String line;
            while((line = br.readLine()) != null) {
                bw.write(line);        //没有换行的写入
                //bw.newLine();        //回车换行符
                //bw.write("\r\n");        //与newLine方法作用相同
            }
            
            br.close();
            bw.close();        //若不关流,会无法写到aaa.txt中
        }
      
        public static void demo1() throws FileNotFoundException, IOException {
            BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));
            String line;
            
            while((line = br.readLine()) != null) {        //返回String类,遇到null说明读取完了,可以整行整行读取
                System.out.println(line);    //一定要println,若是print,拷贝文字的时候不会换行
            }
            
            br.close();
        }
      

21.09_IO流(将文本反转)

  • 将一个文本文档上的文本反转,第一行和倒数第一行交换,第二行和倒数第二行交换

      public static void main(String[] args) throws IOException {
          //改写后是尽量晚开早关
          // 1,创建输入输出流对象
          BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));
          
          //2,创建集合对象
          ArrayList<String> list = new ArrayList<>();
          
          //3,将读到的数据存储在集合中
          String line;
          while((line = br.readLine()) != null) {
              list.add(line);
          }
          br.close();            //关流,造关
          
          //4,倒着遍历集合将数据写到文件上
          BufferedWriter bw = new BufferedWriter(new FileWriter("revzzz.txt"));        //晚开
          for(int i = list.size() - 1; i >= 0; i--) {
              bw.write(list.get(i));
              bw.newLine();        //换行符
          }
          //5,关流
          
          bw.close();
      }
    

21.10_IO流(LineNumberReader)

  • LineNumberReader是BufferedReader的子类, 具有相同的功能, 并且可以统计行号

    • 调用getLineNumber()方法可以获取当前行号
    • 调用setLineNumber()方法可以设置当前行号
  •   LineNumberReader lnr = new LineNumberReader(new FileReader("aaa.txt"));
      String line;
      lnr.setLineNumber(100);            //设置行号,运行后的行号从101开始
      while((line = lnr.readLine()) != null) {
          System.out.println(lnr.getLineNumber() + ":" + line);//获取行号,100+1,100+2,,,
      }
      
      lnr.close(); 
    

21.11_IO流(装饰设计模式)

        /**
         * @param args
         * 装饰设计模式的好处是:
             * 耦合性不强(比继承(子父类)来说),被装饰的类的变化与装饰类的变化无关
         */
    public static void main(String[] args) {
        HeiMaStudent hms = new HeiMaStudent(new Student());
        hms.code();
    }
    
    interface Coder {
        public void code();
    }
    
    class Student implements Coder {
    
        @Override
        public void code() {
            System.out.println("javase");
            System.out.println("javaweb");
        }
        
    }
    
    class HeiMaStudent implements Coder {
        //1,获取被装饰类的引用
        private Student s;        //获取学生引用
        
        //2,在构造方法中传入被装饰类的对象
        public HeiMaStudent(Student s) {
            this.s = s;
        }
    
        //3,对原有的功能进行升级
        @Override
        public void code() {
            s.code();
            System.out.println("ssh");
            System.out.println("数据库");
            System.out.println("大数据");
            System.out.println("...");
        }

21.12_IO流(使用指定的码表读写字符)

FileReader是使用默认码表读取文件, 如果需要使用指定码表读取, 那么可以使用InputStreamReader(是字节流通向字符流的桥梁:字符流,接受参数:字节流,编码表)

FileWriter是使用默认码表写出文件, 如果需要使用指定码表写出, 那么可以使用OutputStreamWriter(是字符流通向字节流的桥梁:字符流,接受参数:字节流,编码表)

  •   /**
       * @param args
       * @throws IOException 
       */
      public static void main(String[] args) throws IOException {
          //demo1();
          //demo2();
          BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream("utf-8.txt"), "utf-8"));        //更高效的读
    
          BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(new FileOutputStream("gbk.txt"), "gbk"));        //更高效的写
    
          int c;
          while((c = br.read()) != -1) {
              bw.write(c);
          }
          
          br.close();
          bw.close();
      }
    
      public static void demo2() throws UnsupportedEncodingException,
              FileNotFoundException, IOException {
          InputStreamReader isr = new InputStreamReader(new FileInputStream("utf-8.txt"), "uTf-8");        //指定码表读字符
          OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("gbk.txt"), "gbk");        //指定码表写字符
          
          int c;
          while((c = isr.read()) != -1) {
              osw.write(c);
          }
          
          isr.close();
          osw.close();
      }
    
      public static void demo1() throws FileNotFoundException, IOException {
          //用默认编码表读写,出现乱码
          FileReader fr = new FileReader("utf-8.txt");    //在默认utf-8格式下读,一个中文3个字节                
          FileWriter fw = new FileWriter("gbk.txt");        //用默认gbk格式写,一个中文2个字节
          
          int c;
          while((c = fr.read()) != -1) {
              fw.write(c);
          }
          
          fr.close();
          fw.close();
      }
    

21.13_IO流(转换流图解)

画图分析转换流


utf-8.txt——>FilelnputStream(字节流读取)——>InputStreamReader(字节流,编码表):字节通向字符的桥梁,通过指定的编码表将字节转换为字符——>BufferedReader,装入缓冲区提高效率——>JAVA程序——>BufferedWriter——>OutputStreamWriter(字节流,编码表):字符通向字节的桥梁,通过指定的编码表将字符转换为字节——>FileOutputStream——>gbk.txt

21.14_IO流(获取文本上字符出现的次数)

  • 获取一个文本上每个字符出现的次数,将结果写在times.txt上

      public static void main(String[] args) throws IOException {
          //1,创建带缓冲的输入流对象
          BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));
    
          //2,创建双列集合对象TreeMap
          TreeMap<Character, Integer> tm = new TreeMap<>();
    
          //3,将读到的字符存储在双列集合中,存储的时候要做判断,如果不包含这个键,就将键和1存储,如果包含这个键,就将该键和值加1存储
          int ch;
          while((ch = br.read()) != -1) {
              char c = (char)ch;        //强制类型转换
              /*if(!tm.containsKey(c)) {
                  tm.put(c, 1);
              }else {
                  tm.put(c, tm.get(c) + 1);    //通过键获取值
              }*/
              tm.put(c, !tm.containsKey(c) ? 1 : tm.get(c) + 1);
          }
    
          //4,关闭输入流
          br.close();        //早关
    
          //5,创建输出流对象
          BufferedWriter bw = new BufferedWriter(new FileWriter("times.txt"));        //晚开
    
          //6,遍历集合将集合中的内容写到times.txt中
          for(Character key : tm.keySet()) {
              switch (key) {
              case '\t':
                  bw.write("\\t" + "=" + tm.get(key));     
                  break;
              case '\n':
                  bw.write("\\n" + "=" + tm.get(key)); 
                  break;
              case '\r':
                  bw.write("\\r" + "=" + tm.get(key)); 
                  break;
              default:
                  bw.write(key + "=" + tm.get(key));         //写出键和值
                  break;
              }
              bw.newLine();        //换行
          }
    
          //7,关闭输出流
          bw.close();
      }
    

21.15_IO流(试用版软件)

  • 当我们下载一个试用版软件,没有购买正版的时候,每执行一次就会提醒我们还有多少次使用机会用学过的IO流知识,模拟试用版软件,试用10次机会,执行一次就提示一次您还有几次机会,如果次数到了提示请购买正版

      public static void main(String[] args) throws IOException {
          //1,创建带缓冲的输入流对象,因为要使用readLine方法,可以保证数据的原样性
          BufferedReader br = new BufferedReader(new FileReader("config.txt"));
    
          //2,将读到的字符串转换为int数
          String line = br.readLine();
          int times = Integer.parseInt(line);        //将数字字符串转换为数字
    
          //3,对int数进行判断,如果大于0,就将其--写回去,如果不大于0,就提示请购买正版
          if(times > 0) {
    
              //4,在if判断中要将--的结果打印,并将结果通过输出流写到文件上
              System.out.println("您还有" + times-- + "次机会");
              FileWriter fw = new FileWriter("config.txt");    //此处清空没关系,因为已写入缓存中
              fw.write(times + "");    //以字符串形式写回去,加空串
              fw.close();        //一定要记得关流,否则之前写入缓存的数据再次运行时会被清空
          }else {
              System.out.println("您的试用次数已到,请购买正版");
          }
          
          br.close();        //关闭流
      }
    

21.16_File类(递归)

  • 5的阶乘

      /**
       * @param args
       * 递归:方法自己调用自己
       * 5!
       * 5 * 4 * 3 * 2 * 1
       * 
       * 5 * fun(4)(代表4!)
       *         4 * fun(3)(代表3!)
       *                 3 * fun(2)(代表2!) 
       *                         2 * fun(1)(代表1!)
       * 递归的弊端:不能调用次数过多,容易导致栈内存溢出
       * 递归的好处:不用知道循环次数
       * 
       * 构造方法是否可以递归调用?
           * 构造方法不能使用递归调用,无法终止
       * 
       * 递归调用是否必须有返回值?
           * 不一定(可以有,也可以没有)
       */
    

21.17_File类(练习)

  • 需求:从键盘输入接收一个文件夹路径,打印出该文件夹下所有的.java文件名

      /**
       * 需求:从键盘输入接收一个文件夹路径,打印出该文件夹下所有的.java文件名
       * 
       * 分析:
       * 从键盘接收一个文件夹路径
       * 1,如果录入的是不存在,给与提示
       * 2,如果录入的是文件路径,给与提示
       * 3,如果是文件夹路径,直接返回
       * 
       * 打印出该文件夹下所有的.java文件名
       * 1,获取到该文件夹路径下的所有的文件和文件夹,存储在File数组中
       * 2,遍历数组,对每一个文件或文件夹做判断
       * 3,如果是文件,并且后缀是.java的,就打印
       * 4,如果是文件夹,就递归调用
       */
      public static void main(String[] args) {
          File dir = getDir();
          printJavaFile(dir);
      }
    
      /*
       * 获取键盘录入的文件夹路径
       * 1,返回值类型File
       * 2,不需要有参数
       */
      public static File getDir() {
          Scanner sc = new Scanner(System.in);    //创建键盘录入对象
          System.out.println("请输入一个文件夹路径");
          while(true) {
              String line = sc.nextLine();    //将键盘录入的文件夹路径存储
              File dir = new File(line);        //封装成File对象
              if(!dir.exists()) {
                  System.out.println("您录入的文件夹路径不存在,请重新录入");
              }else if(dir.isFile()) {
                  System.out.println("您录入的是文件路径,请重新录入文件夹路径");
              }else {
                  return dir;
              }
          }
      }
      /*
       * 获取文件夹路径下的所.java文件
       * 1,返回值类型 void
       * 2,参数列表File dir
       */
      public static void printJavaFile(File dir) {
          //1,获取到该文件夹路径下的所有的文件和文件夹,存储在File数组中
          File[] subFiles = dir.listFiles();
    
          //2,遍历数组,对每一个文件或文件夹做判断
          for (File subFile : subFiles) {
    
              //3,如果是文件,并且后缀是.java的,就打印
              if(subFile.isFile() && subFile.getName().endsWith(".java")) {
                  System.out.println(subFile);
    
              //4,如果是文件夹,就递归调用
              }else if (subFile.isDirectory()){
                  printJavaFile(subFile);
              }
          }
      }
    

21.18_IO流(总结)

  • 1.会用BufferedReader读取GBK码表和UTF-8码表的字符
  • 2.会用BufferedWriter写出字符到GBK码表和UTF-8码表的文件中
  • 3.会使用BufferedReader从键盘读取一行
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值