2019独角兽企业重金招聘Python工程师标准>>>
21.01_IO流(字符流FileReader)
- 1.字符流是什么
- 字符流是可以直接读写字符的IO流
- 字符流读取字符, 就要先读取到字节数据, 然后转为字符. 如果要写出字符, 需要把字符转为字节再写出.
- 2.FileReader
- FileReader类的read()方法可以按照字符大小读取
-
FileReader fr = new FileReader("aaa.txt"); //创建输入流对象,关联aaa.txt int ch; while((ch = fr.read()) != -1) { //将读到的字符赋值给chSystem.out.println((char)ch); //将读到的字符强转后打印 }fr.close(); //关流
package com.heima.chario;import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;public class Demo1_FileReader {/*** @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) { //通过项目默认的码表一次读取一个字符System.out.print((char)c);}fr.close();}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();}}
21.02_IO流(字符流FileWriter)
-
FileWriter类的write()方法可以自动把字符转为字节写出
FileWriter fw = new FileWriter("aaa.txt"); fw.write("aaa"); fw.close();
package com.heima.chario;import java.io.FileWriter;
import java.io.IOException;public class Demo2_FileWriter {/*** @param args* @throws IOException */public static void main(String[] args) throws IOException {FileWriter fw = new FileWriter("yyy.txt");fw.write("大家好,基础班快接近尾声了,大家要努力,要坚持!!!");fw.write(97);fw.close();}}
21.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();
package com.heima.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 Demo3_Copy {/*** @param args* @throws IOException */public static void main(String[] args) throws IOException {//demo1();//demo2(); //demo3();BufferedReader br = new BufferedReader(new FileReader("xxx.txt"));BufferedWriter bw = new BufferedWriter(new FileWriter("yyy.txt"));int c;while((c = br.read()) != -1) {bw.write(c);}br.close();bw.close();}public static void demo3() throws FileNotFoundException, IOException {FileReader fr = new FileReader("xxx.txt");FileWriter fw = new FileWriter("yyy.txt");char[] arr = new char[1024];int len;while((len = fr.read(arr)) != -1) { //将文件上的数据读取到字符数组中fw.write(arr,0,len); //将字符数组中的数据写到文件上}fr.close();fw.close();}public static void demo2() throws FileNotFoundException, IOException {//字符流不能拷贝纯文本的文件FileReader fr = new FileReader("双元.jpg");FileWriter fw = new FileWriter("copy.jpg");int c;while((c = fr.read()) != -1) {fw.write(c);}fr.close();fw.close();}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) {fw.write(c);}fr.close();fw.close(); //Writer类中有一个2k的小缓冲区,如果不关流,就会将内容写到缓冲区里,关流会将缓冲区内容刷新,再关闭}}
21.04_IO流(什么情况下使用字符流)
- 字符流也可以拷贝文本文件, 但不推荐使用. 因为读取时会把字节转为字符, 写出时还要把字符转回字节.
- 程序需要读取一段文本, 或者需要写出一段文本的时候可以使用字符流
- 读取的时候是按照字符的大小读取的,不会出现半个中文
- 写出的时候可以直接将字符串写出,不用转换为字节数组
21.05_IO流(字符流是否可以拷贝非纯文本的文件)
- 不可以拷贝非纯文本的文件
- 因为在读的时候会将字节转换为字符,在转换过程中,可能找不到对应的字符,就会用?代替,写出的时候会将字符转换成字节写出去
- 如果是?,直接写出,这样写出之后的文件就乱了,看不了了
21.06_IO流(自定义字符数组的拷贝)
-
FileReader fr = new FileReader("aaa.txt"); //创建字符输入流,关联aaa.txt FileWriter 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();
21.07_IO流(带缓冲的字符流)
- BufferedReader的read()方法读取字符时会一次读取若干字符到缓冲区, 然后逐个返回给程序, 降低读取文件的次数, 提高效率
- BufferedWriter的write()方法写出字符时会先写到缓冲区, 缓冲区写满时才会写到文件, 降低写文件的次数, 提高效率
-
BufferedReader br = new BufferedReader(new FileReader("aaa.txt")); //创建字符输入流对象,关联aaa.txt BufferedWriter bw = new BufferedWriter(new FileWriter("bbb.txt")); //创建字符输出流对象,关联bbb.txtint ch; while((ch = br.read()) != -1) { //read一次,会先将缓冲区读满,从缓冲去中一个一个的返给临时变量chbw.write(ch); //write一次,是将数据装到字符数组,装满后再一起写出去 }br.close(); //关流 bw.close();
21.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();
package com.heima.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_Buffered {/*** @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"));BufferedWriter bw = new BufferedWriter(new FileWriter("aaa.txt"));String line;while((line = br.readLine()) != null) {bw.write(line);//bw.newLine(); //写出回车换行符//bw.write("\r\n");}br.close();bw.close();}public static void demo1() throws FileNotFoundException, IOException {BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));String line;while((line = br.readLine()) != null) {System.out.println(line);}br.close();}}
21.09_IO流(将文本反转)
- 将一个文本文档上的文本反转,第一行和倒数第一行交换,第二行和倒数第二行交换
package com.heima.test;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.ArrayList;public class Test1 {/*** @param args* 将一个文本文档上的文本反转,第一行和倒数第一行交换,第二行和倒数第二行交换* * 分析:* 1,创建输入输出流对象* 2,创建集合对象* 3,将读到的数据存储在集合中* 4,倒着遍历集合将数据写到文件上* 5,关流* @throws IOException * * 注意事项:* 流对象尽量晚开早关*/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); //设置行号 while((line = lnr.readLine()) != null) {System.out.println(lnr.getLineNumber() + ":" + line);//获取行号 }lnr.close();
package com.heima.chario;import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.io.LineNumberReader;public class Demo5_LineNumberReader {/*** @param args* @throws IOException */public static void main(String[] args) throws IOException {LineNumberReader lnr = new LineNumberReader(new FileReader("zzz.txt"));String line;lnr.setLineNumber(100);while((line = lnr.readLine()) != null) {System.out.println(lnr.getLineNumber() + ":" + line);}lnr.close();}}
21.11_IO流(装饰设计模式)
-
interface Coder {public void code(); }class Student implements Coder {@Overridepublic void code() {System.out.println("javase");System.out.println("javaweb");}}class HeiMaStudent implements Coder {private Student s; //获取到被包装的类的引用public HeiMaStudent(Student s) { //通过构造函数创建对象的时候,传入被包装的对象this.s = s;}@Overridepublic void code() { //对其原有功能进行升级s.code();System.out.println("数据库");System.out.println("ssh");System.out.println("安卓");System.out.println(".....");}}
package com.heima.chario;public class Demo6_Wrap {/*** @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 {@Overridepublic 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,对原有的功能进行升级@Overridepublic void code() {s.code();System.out.println("ssh");System.out.println("数据库");System.out.println("大数据");System.out.println("...");}
}
21.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();
package com.heima.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_TransIO {/*** @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"); FileWriter fw = new FileWriter("gbk.txt");int c;while((c = fr.read()) != -1) {fw.write(c);}fr.close();fw.close();}}
21.13_IO流(转换流图解)
- 画图分析转换流
21.14_IO流(获取文本上字符出现的次数)
- 获取一个文本上每个字符出现的次数,将结果写在times.txt上
package com.heima.test;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.HashMap;public class Test2 {/*** @param args* 获取一个文本上每个字符出现的次数,将结果写在times.txt上* * 1,创建带缓冲区的输入流对象* 2,创建双列集合对象,目的是把字符当作键,把字符出现的次数当作值* 3,通过读取不断向集合中存储,存储的时候要判断,如果不包含这个键就将键和值为1存储,如果包含就将键和值加1存储* 4,关闭输入流* 5,创建输出流对象* 6,将结果写出* 7,关闭输出流* @throws IOException */public static void main(String[] args) throws IOException {//1,创建带缓冲区的输入流对象BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));//2,创建双列集合对象,目的是把字符当作键,把字符出现的次数当作值HashMap<Character, Integer> hm = new HashMap<>();//3,通过读取不断向集合中存储,存储的时候要判断,如果不包含这个键就将键和值为1存储,如果包含就将键和值加1存储int c;while((c = br.read()) != -1) {char ch = (char)c;/*if(!hm.containsKey(ch)) {hm.put(ch, 1);}else {hm.put(ch, hm.get(ch) + 1);}*/hm.put(ch, !hm.containsKey(ch)? 1 : hm.get(ch) + 1);}//4,关闭输入流br.close();//5,创建输出流对象BufferedWriter bw = new BufferedWriter(new FileWriter("times.txt"));//6,将结果写出for (Character key : hm.keySet()) {bw.write(key + "=" + hm.get(key));}bw.close();}}
package com.heima.test;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 Test3 {/*** 获取一个文本上每个字符出现的次数,将结果写在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,创建双列集合对象TreeMapTreeMap<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次机会,执行一次就提示一次您还有几次机会,如果次数到了提示请购买正版
package com.heima.test;import java.io.BufferedReader;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;public class Test4 {/*** 当我们下载一个试用版软件,没有购买正版的时候,每执行一次就会提醒我们还有多少次使用机会用学过的IO流知识,模拟试用版软件,* 试用10次机会,执行一次就提示一次您还有几次机会,如果次数到了提示请购买正版* @throws IOException * 分析:* 1,创建带缓冲的输入流对象,因为要使用readLine方法,可以保证数据的原样性* 2,将读到的字符串转换为int数* 3,对int数进行判断,如果大于0,就将其--写回去,如果不大于0,就提示请购买正版* 4,在if判断中要将--的结果打印,并将结果通过输出流写到文件上*/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的阶乘
package com.heima.chario;public class Demo8_Digui {/*** @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);}}
}
21.17_File类(练习)
- 需求:从键盘输入接收一个文件夹路径,打印出该文件夹下所有的.java文件名
package com.heima.test;import java.io.File;
import java.io.FileReader;
import java.util.Scanner;public class Test5 {/*** 需求:从键盘输入接收一个文件夹路径,打印出该文件夹下所有的.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从键盘读取一行