22 IO流-字符流、缓冲字符流、递归

来源:互联网 发布:不规则体积计算软件 编辑:程序博客网 时间:2024/05/22 15:58

byte字节 8位 char是两个字节,为字符 16位

01_IO流(字符流FileReader)

  • 1.字符流是什么
    • 字符流是可以直接读写字符的IO流
    • 字符流读取字符, 就要先读取到字节数据, 然后转为字符. 如果要写出字符, 需要把字符转为字节再写出.
  • 2.FileReader
    • FileReader类的read()方法可以按照字符大小读取
  • FileReader fr = new FileReader("aaa.txt");              //创建输入流对象,关联aaa.txtint ch;while((ch = fr.read()) != -1) {                         //将读到的字符赋值给ch    System.out.println((char)ch);                       //将读到的字符强转后打印}fr.close();                                             //关流 

02_IO流(字符流FileWriter)

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

    FileWriter fw = new FileWriter("aaa.txt");fw.write("aaa");fw.close();

03_IO流(字符流的拷贝)

FileReader fr = new FileReader("a.txt");FileWriter fw = new FileWriter("b.txt");int ch;while((ch = fr.read()) != -1) {    fw.write(ch);}fr.close();fw.close(); write类原码中有一个char[1024]也就是2M的缓冲区,fw如果不关闭或者刷新的话,复制就会失败

04_IO流(什么情况下使用字符流;字符流和字节流的区别)

  • 字符流也可以拷贝文本文件, 但不推荐使用. 因为读取时会把字节转为字符, 写出时还要把字符转回字节.
  • 程序需要读取一段文本, 或者需要写出一段文本的时候可以使用字符流
  • 读取的时候是按照字符的大小读取的,不会出现半个中文
  • 写出的时候可以直接将字符串写出,不用转换为字节数组
    字符流和字节流的区别
    转载自http://www.importnew.com/11537.html
    InputStream和Reader都是抽象类,并不直接地从文件或者套接字(socket)中读取数据。然而,它们之间的主要差别在于:InputStream用于读取二进制数据(字节流方式,译者注),Reader用于读取文本数据(字符流方式,译者注),准确地说,Unicode字符。那么,二进制数据和文本数据的区别是什么呢?当然,所有读取的东西本质上是字节,然后需要一套字符编码方案,把字节转换成文本。Reader类使用字符编码来解码字节,并返回字符给调用者。Reader类要么使用运行Java程序平台的默认字符编码,要么使用Charset对象或者String类型的字符编码名称,如“UTF-8”。尽管它是一个最简单的概念,当读取文本文件或从套接字中读取文本数据时,很多Java开发者会因没有指定字符编码而犯错。记住,如果你没有指定正确的编码,或者你的程序没有使用的协议中已存在的字符编码,如HTML的 “Content-Type(内容类型)”、XML文件头指定的编码,你可能无法正确地读取的所有数据。一些不是默认编码呈现的字符,可能变成“?”或小方格。一旦你知道stream和reader之间的根本区别,理解FileInputStream和FileReader之间的差异就很容易了。既可以让你从文件中读取数据,然而FileInputStream用于读取二进制数据,FileReader用来读取字符数据。
    这里写图片描述

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

  • 不可以拷贝非纯文本的文件
  • 因为在读的时候会将字节转换为字符,在转换过程中,可能找不到对应的字符,就会用?代替,写出的时候会将字符转换成字节写出去
  • 如果是?,直接写出,这样写出之后的文件就乱了,看不了了
    一句话,就是图片里面的有些字节在编码表里面没有,编码表不是任何字节都能成功转码

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

  • FileReader fr = new FileReader("aaa.txt");          //创建字符输入流,关联aaa.txtFileWriter fw = new FileWriter("bbb.txt");          //创建字符输出流,关联bbb.txtint len;char[] arr = new char[1024*8];                      //创建字符数组while((len = fr.read(arr)) != -1) {                 //将数据读到字符数组中    fw.write(arr, 0, len);                          //从字符数组将数据写到文件上}fr.close();                                         //关流释放资源fw.close(); 

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

  • BufferedReader的read()方法读取字符时会一次读取若干字符到缓冲区, 然后逐个返回给程序, 降低读取文件的次数, 提高效率
  • BufferedWriter的write()方法写出字符时会先写到缓冲区, 缓冲区写满时才会写到文件, 降低写文件的次数, 提高效率
  • BufferedReader br = new BufferedReader(new FileReader("aaa.txt"));  //创建字符输入流对象,关联aaa.txtBufferedWriter bw = new BufferedWriter(new FileWriter("bbb.txt"));  //创建字符输出流对象,关联bbb.txtint ch;             while((ch = br.read()) != -1) {     //read一次,会先将缓冲区读满,从缓冲去中一个一个的返给临时变量ch    bw.write(ch);                   //write一次,是将数据装到字符数组,装满后再一起写出去}br.close();                         //关流bw.close();  

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

  • BufferedReader的readLine()方法可以读取一行字符(不包含换行符号)
  • BufferedWriter的newLine()可以输出一个跨平台的换行符号”\r\n”
  • BufferedReader br = new BufferedReader(new FileReader("aaa.txt"));BufferedWriter bw = new BufferedWriter(new FileWriter("bbb.txt"));String line;while((line = br.readLine()) != null) {    bw.write(line);    //bw.write("\r\n");                 //只支持windows系统    bw.newLine();                       //跨平台的}br.close();bw.close(); 

09_IO流(将文本反转)

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

public class Reverse {
public static void main(String[] args) throws IOException {
FileReader fr=new FileReader(“d://tt.txt”);

BufferedReader bf=new BufferedReader(fr);ArrayList al=new ArrayList();String a;while((a=bf.readLine())!=null){    al.add(a);      }bf.close();FileWriter fw=new FileWriter("d://tt.txt");BufferedWriter bw=new BufferedWriter(fw);for (int i = al.size()-1; i >=0; i--) {    bw.write((String)al.get(i));    bw.newLine();}   bw.close();

}
}
注意:reader流要提前关闭。不然文件写不进去

10_IO流(LineNumberReader)

  • LineNumberReader是BufferedReader的子类, 具有相同的功能, 并且可以统计行号
    • 调用getLineNumber()方法可以获取当前行号
    • 调用setLineNumber()方法可以设置当前行号
  • LineNumberReader lnr = new LineNumberReader(new FileReader("aaa.txt"));String line;lnr.setLineNumber(100);                                 //设置行号while((line = lnr.readLine()) != null) {    System.out.println(lnr.getLineNumber() + ":" + line);//获取行号}lnr.close(); 

11_IO流(装饰设计模式)

  • interface Coder {    public void code();}class Student implements Coder {    @Override    public void code() {        System.out.println("javase");        System.out.println("javaweb");    }}class MaStudent implements Coder {    private Student s;                      //获取到被包装的类的引用    public ItcastStudent(Student s) {       //通过构造函数创建对象的时候,传入被包装的对象        this.s = s;    }    @Override    public void code() {                    //对其原有功能进行升级        s.code();        System.out.println("数据库");        System.out.println("ssh");        System.out.println("安卓");        System.out.println(".....");    }} 

    装饰设计模式的好处:
    被装饰的类与装饰的类的耦合性降低,被装饰的类Student的变化与装饰的类MaStudent 无关

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

  • FileReader是使用默认码表读取文件, 如果需要使用指定码表读取, 那么可以使用InputStreamReader(字节流,编码表)
  • FileWriter是使用默认码表写出文件, 如果需要使用指定码表写出, 那么可以使用OutputStreamWriter(字节流,编码表)
  • 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 ch;while((ch = br.read()) != -1) {    bw.write(ch);}br.close();bw.close();

13_IO流(转换流图解)

  • 画图分析转换流
    这里写图片描述

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

  • 获取一个文本上每个字符出现的次数,将结果写在times.txt上
    * 在Java7之前,switch只能支持 byte、short、char、int或者其对应的封装类以及Enum类型。在Java7中,也支持了String类型。
  • 两种方法:
  • 第一种是自己的方法:由key直接判定value是否为null?1:+1;
  • switch里面把char转换的字符串,后来学习才知道,传入char也可以
public static void main(String[] args) throws IOException {        // 1.输入流读文档        // 2.把读到的数据放到map集合里面,又重复的就加一        // 3,用readline来读取,写在timesresult        FileReader fr = new FileReader("timesresult.txt");        int a;        HashMap<Character, Integer> hm = new HashMap<>();        while ((a = fr.read()) != -1) {            hm.put((char) a, (hm.get((char) a) == null) ? 1 : hm.get((char) a) + 1);        }        BufferedWriter  bw=new BufferedWriter(new FileWriter("times1.txt"));        for (char s : hm.keySet()) {            String w=String.valueOf(s);            switch (w) {            case "\t":                System.out.println("//t" + " " + hm.get(s));                bw.write("//t" + " " + hm.get(s));                break;            case "\r":                System.out.println("//r" + " " + hm.get(s));                bw.write("//r" + " " + hm.get(s));                break;            case "\n":                System.out.println("//n" + " " + hm.get(s));                bw.write("//n" + " " + hm.get(s));                break;            default:                System.out.println(s + " " + hm.get(s));                bw.write(s + " " + hm.get(s));                break;            }            bw.newLine();        }        bw.close();    }

第二种方法是把用集合的containsKey()方法来判定是否已经存在,存在就value=+1

public static void main(String[] args) throws IOException {        //1,创建带缓冲的输入流对象        BufferedReader br = new BufferedReader(new FileReader("timesresult.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();    }

15_IO流(试用版软件)

  • 当我们下载一个试用版软件,没有购买正版的时候,每执行一次就会提醒我们还有多少次使用机会用学过的IO流知识,模拟试用版软件,试用10次机会,执行一次就提示一次您还有几次机会,如果次数到了提示请购买正版
    我的方法:
public static void main(String[] args) throws IOException {    BufferedReader br=new BufferedReader(new FileReader("a.txt"));    String a;    int b=0;    while((a=br.readLine())!=null){        b=Integer.parseInt(a);        System.out.println("你还有"+b--+"天试用期");    }    br.close();    FileOutputStream fis=new FileOutputStream("a.txt");    fis.write(String.valueOf(b).getBytes());    fis.close();    if(b==-1){        System.out.println("试用期已经过了,你要注册以后才可以继续使用");    }}}
或者用字符输出流FileWriter fw=new FileWriter("a.txt");    fw.write(String.valueOf(b));    fw.close();

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

自己写的:中间有数组的递归来遍历

public static void main(String[] args) {        System.out.println(multiply(10));        int arr[]={1,4,2,6,8,33,8,6,3,3};        arrayview(arr,5);    }    public static int multiply(int num){        if(num==1)            return 1;        else             return num*multiply(num-1);    }    public static void arrayview(int arr[],int num){        if(num<0)            return;        else             System.out.println(arr[num]);        arrayview(arr,num-1);    }

17_File类(练习)

  • 需求:从键盘输入接收一个文件夹路径,打印出该文件夹下所有的.java文件名
public class Test5 {    /**     *      * 分析:     * 从键盘接收一个文件夹路径     * 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);            }        }    }}
0 0