2019-05-30 Java学习日记之IO(字符流)&字符流其他内容&递归

IO(字符流)

 字符流FileReader:

1、字符流是什么?

字符流是可以直接读写字符的IO流

字符流读取字符,就要先读取到字节数据,然后转为字符。如果要写出字符,需要吧字符转为字节再写出

2、FileReader

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

package com.chario;

import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;

public class Demo1 {

    public static void main(String[] args) throws IOException {
//        demo1();
        FileReader fr = new FileReader("xxx.txt");
        int c;
        while((c = fr.read()) != -1){                //通过项目默认的码表一次读取一个字符
            System.out.print((char)c);
        }
    }

    public static void demo1() throws FileNotFoundException, IOException {
        FileReader fr = new FileReader("xxx.txt");
        int x = fr.read();
        System.out.println(x);
        char c = (char)x;
        System.out.println(c);
        fr.close();
    }

}

字符流FileWriter:

FileWriter类的writer()方法可以自动吧字符转为字节写出

package com.chario;

import java.io.FileWriter;
import java.io.IOException;

public class Demo2 {
    public static void main(String[] args) throws IOException {
        FileWriter fw = new FileWriter("yyy.txt");
        fw.write("努力,坚持");
        fw.write(99);
        fw.close();
    }
}

字符流的拷贝:

package com.chario;

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

public class Demo3 {
    public static void main(String[] args) throws IOException {
        FileReader fr1 = new FileReader("xxx.txt");
        FileWriter fw1 = new FileWriter("zzz.txt");
        
        int c;
        while ((c = fr1.read()) != -1) {
            fw1.write(c);
        }
         fr1.close();
         fw1.close();        //Writer类中有一个2k的小缓冲区,如果不关流,就会将内容写到缓冲区里,关流会将缓冲区内容刷新,再关闭
    }
}

什么情况下使用字符流:

字符流也可以拷贝文本文件,但不推荐使用。因为读取时会把字节转为字符,写出时还有把字符转回字节。

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

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

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

字符流是否可以拷贝非纯文本的文件:

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

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

写出的时候会将字符转换成字节写出去

如果是?,直接写出,这样写出之后的文件就乱了,不用转换为字节数组

自定义字符数组的拷贝:

public static void demo2() throws FileNotFoundException, IOException {
        FileReader fr1 = new FileReader("xxx.txt");
        FileWriter fw1 = new FileWriter("yyy.txt");
        
        char[] arr = new char[1024];
        int len;
        while ((len = fr1.read(arr)) != -1) {    //将文件上的数据读取到字符数组中    
            fw1.write(arr,0,len);                //将字符数组中的数据写到文件上
        }
        
        fr1.close();
        fw1.close();
    }

带缓冲的字符流:

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

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

public static void demo3() throws FileNotFoundException, IOException {
        BufferedReader bReader = new BufferedReader(new FileReader("xxx.txt"));
        BufferedWriter bWriter = new BufferedWriter(new FileWriter("yyy.txt"));
        
        int c; 
        while ((c = bReader.read()) != -1) {
            bWriter.write(c);
        }
        
        bReader.close();
        bWriter.close();
    }

readLine()和newLine()方法:

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

BufferedWriter的newLine()可以输出一个跨平台的换行符号" ";

package com.chario;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;

public class Demo4 {
    /**
     * 带缓冲区的流中的特殊方法
     * readLine()
     * newLine()
     * 
     * newLine()
的区别
     * newLine()是跨平台的方法
     * 
值支持的是windows系统
     * 
     */
    public static void main(String[] args) throws IOException {
//        demo1();
        BufferedReader bReader = new BufferedReader(new FileReader("zzz.txt"));
        BufferedWriter bWriter = new BufferedWriter(new FileWriter("aaa.txt"));
        
        String line;
        while((line = bReader.readLine()) != null){
            bWriter.write(line);
            //bWriter.newLine();             //写出回车换行符
            bWriter.write("
");
        }
        
        bReader.close();
        bWriter.close();
    }

    public static void demo1() throws FileNotFoundException, IOException {
        BufferedReader bReader = new BufferedReader(new FileReader("zzz.txt"));
        String line;
        
        while ((line = bReader.readLine()) != null) {
            System.out.println(line);
        }
        bReader.close();
    }
}

将文本反转:

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

package com.chario;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.util.ArrayList;

public class Test1 {
    /**
     * 需求:将一个文本文档上的文本反转,第一行和倒数第一行交换,第二行和倒数第二行交换
     * 
     * 分析:
     * 1、创建输入输出流对象
     * 2、创建集合对象
     * 3、将读到的数据存储在集合中
     * 4、倒着遍历集合将数据写到文件上
     * 5、关流
     * 
     * 注意事项:流对象尽量晚开早关
     * @throws IOException 
     * 
     */
    public static void main(String[] args) throws IOException {
        //1、创建输入输出流对象
        BufferedReader bReader = new BufferedReader(new FileReader("zzz.txt"));
        
        //2、创建集合对象
        ArrayList<String> list = new ArrayList<>();
        //3、将读到的数据存储在集合中
        String line;
        while((line = bReader.readLine()) != null){
            list.add(line);
        }
        bReader.close();            //关流
        //4、倒着遍历集合将数据写到文件上
        BufferedWriter bWriter = new BufferedWriter(new FileWriter("revzzz.txt"));
        for (int i = list.size() - 1; i >= 0; i--) {
            bWriter.write(list.get(i));
            bWriter.newLine();
        }
        //5、关流
        bWriter.close();
    }
}

LineNumberReader:

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

调用getLineNumber()方法可以获取当前行号

调用setLineNumber()方法可以设置当前行号

package com.chario;

import java.io.FileReader;
import java.io.IOException;
import java.io.LineNumberReader;

public class Demo5 {
    public static void main(String[] args) throws IOException {
        LineNumberReader lnr1 = new LineNumberReader(new FileReader("zzz.txt"));
        
        String line;
        lnr1.setLineNumber(100);
        
        while((line = lnr1.readLine()) != null){
            System.out.println(lnr1.getLineNumber() + ":" + line);
        }
        lnr1.close();
    }
}

装饰设计模式:

package com.chario;

public class Demo6 {
    /**
     *装饰设计模式的好处是:
     *耦合性不强,被装饰类的变化与装饰类的变化无关
     */
    public static void main(String[] args) {
        HeiMaStudent heiMaStudent = new HeiMaStudent(new Student());
        heiMaStudent.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("...");
    }
}

使用指定的码表读写字符:

FileReader是使用默认码表读取文件,如果需要使用指定码表读取,那么可以使用InputStreamReader(字节流,编码表)

FileWriter是使用默认码表写出文件,如果需要使用指定码表写出,那么可以使用OutputStreamReader(字节流,编码表)

package com.chario;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.UnsupportedEncodingException;

public class Demo7 {
    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 isr1 = new InputStreamReader(new FileInputStream("utf-8.txt"),"utf-8");    //指定码表读字符
        OutputStreamWriter osw1 = new OutputStreamWriter(new FileOutputStream("gbk.txt"), "gbk");    //指定码表写字符
        
        int c;
        while((c = isr1.read()) != -1) {
            osw1.write(c);
        }
        
        isr1.close();
        osw1.close();
    }

    public static void demo1() throws FileNotFoundException, IOException {
        //用默认的编码表读写,出现乱码
        FileReader fReader = new FileReader("utf-8.txt");
        FileWriter fWriter = new FileWriter("gbk.txt");
        
        int c;
        while((c = fReader.read()) != -1) {
            fWriter.write(c);
        }
        fReader.close();
        fWriter.close();
    }
}

获取文本上字符出现的次数:

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

package com.chario;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.util.TreeMap;

public class Test2 {
    /**
     * 需求:获取一个文本上每个字符出现的次数,将结果写在times.txt上
     * 
     * 分析:
     * 1、创建带缓冲区的输入流对象
     * 2、创建双列集合对象TreeMap
     * 3、将读到的字符存储在双列集合中,存储的时候要做判断,如果不包含这个键,就将键和1存储,如果包含这个键,就将该键和值加1存储
     * 4、关闭输入流
     * 5、创建输出流对象
     * 6、遍历集合将集合中的内容写到times.txt中
     * 7、关闭输出流
     * @throws IOException 
     */
    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 '	':
                bw.write("\t" + "=" + tm.get(key));
                break;
            case '
':
                bw.write("\r" + "=" + tm.get(key));
                break;
            case '
':
                bw.write("\n" + "=" + tm.get(key));
                break;
            default:
                bw.write(key + "=" + tm.get(key));        //写出键和值
                break;
            }
            bw.newLine();
        }
        // 7、关闭输出流
        bw.close();
    }

}

试用版软件:

package com.chario;

import java.io.BufferedReader;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;

public class Test3 {
    /**
     *需求:模拟试用版软件,试用10次机会
     *
     *分析:
     *1、创建带缓冲区的输入流对象,因为要使用readLine方法,可以保证数据的原样性
     *2、将读到的字符串转换为int数
     *3、对int数进行判断,如果大于0,就将其--写回去,如果不大于0,就提示请购买正版
     *4、在if判断中要将--的结果打印,并将结果通过输出流写到文件
     * @throws IOException 
     */
    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();
        
    }

}

递归

5的阶乘:

package com.chario;

public class Demo8 {
    /**
     * 递归的弊端:不能调用次数过多,容易导致栈内存溢出
     * 递归的好处:不用知道循环次数
     * 
     * 构造方法是否可以使用递归调用?
     * 不能
     * 
     * 递归调用是否必须有返回值?
     * 不一定(可以有,也可以没有)
     * 
     */
    public static void main(String[] args) {
        System.out.println(fun(5));
    }
    public static int fun(int num){
        if (num == 1) {
            return 1;
        }else {
            return num * fun(num - 1);
        }
    }

}

Test:

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

package com.chario;

import java.io.File;
import java.util.Scanner;

public class Test4 {
    /**
     * 需求:从键盘输入接收一个文件夹路径,打印出该文件夹下所有的.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);
            }
        }
        
        
    }
}
原文地址:https://www.cnblogs.com/clqbolg/p/10953060.html