IO流(字符流)
字符流
一.字符流是什么
- 字符流是可以直接读取字符的IO流
- 字符流读取字符,就要先读去到字节数据,然后转为字符,如果要写出字符,需要把字符转为字节再写出
FileReader
- FileReader类的read()方法可以按照字符大小读取
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;public class Demo1_FileReader {public static void main(String[] args) throws IOException {demo1();FileReader fr = new FileReader("xxx.txt");int x;while((x = fr.read()) != -1) { //通过项目默认的码表,一次读取一个字符System.out.print((char)x);}fr.close();}public static void demo1() throws FileNotFoundException, IOException {FileReader fr = new FileReader("xxx.txt");//需要创建xxx.txt,如果没有就报错int x = fr.read(); //read方法返回jdk码值System.out.println(x); char c = (char)x; //要想得到字符,要将码值向下强转成对应的字符System.out.println(c);fr.close();}}
FileWriter
- FileWrirer类的write()方法可以自动把字符转为字节写出
import java.io.FileWriter;
import java.io.IOException;public class Demo2_FileWriter {public static void main(String[] args) throws IOException {FileWriter fw = new FileWriter("yyy.txt"); //如果有文件直接写,没有文件就创建一个再写fw.write("zt是傻x");fw.write(122);fw.close();}}
拷贝文件
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;public class Demo3_Copy {public static void main(String[] args) throws IOException {FileReader fr = new FileReader("xxx.txt");FileWriter fw = new FileWriter("yyy.txt");int x;while((x = fr.read()) != -1) {fw.write(x);}fr.close();fw.close();//Write类中有一个2k的小缓冲区,如果不关流,会将文件内容留到缓冲区中,关流会将缓冲区的内容刷新,再关闭}}
一.什么情况下使用字符流
- 字符流也可以拷贝文件,但不推荐使用,因为读取是会酱紫结转为自负,写出时还要将字符转为字节,很麻烦
那什么时候用呢?
- 程序需要只读取一段文本时用字符流,因为读取的时候是按大小读取的,不会出现半个中文
- 程序需要只写出一段文本时用字符流,因为写出的时候可以直接按字符串写出,不用转换为字符数组
二.字符流是否可以拷贝非纯文本文件
- 不可以,因为在读的时候会将字节转换为字符,在转的过程中,可能找不到对应的字符,就用?代替,
写出的时候会将字符转为字节写出去,如果是?,直接写出,这样写出的文件就乱了.
三.自定义字符数组的拷贝
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;public class Demo3_Copy {public static void main(String[] args) throws IOException {//自定义字符数组的拷贝FileReader fr = new FileReader("xxx.txt");FileWriter fw = new FileWriter("yyy.txt");char[] arr = new char[1024];int len;while((len = fr.read()) != -1) { //将文件的数据读取到字符数组中fw.write(arr,0,len); //将字符数组中的数据写到文件中}fr.close();fw.close();}
}
带缓冲区的字符流
一.带缓冲区的流中的常规方法
- BufferedReader的read()方法读取字符时会一次读取若干字符到缓冲区,然后逐个返回给程序,降低读取文件的次数,提高效率
- BufferedWriter的Writer()方法写出字符时会先写到缓冲区,缓冲区写满后才会写到文件,降低写文件的次数,提高效率
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 {public static void main(String[] args) throws IOException {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();}
}
二.带缓冲区的流中的特殊方法
- BufferedReader的readLine()方法可以读取一行字符(不包括换行符号)
- BufferedWriter的newLine()可以输出一个跨平台的换行符号
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 {public static void main(String[] args) throws IOException {//demo1();BufferedReader br = new BufferedReader(new FileReader("xxx.txt"));BufferedWriter bw = new BufferedWriter(new FileWriter("yyy.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("xxx.txt"));String line;//readLine返回一行字符串,用String接收while((line = br.readLine()) != null) {System.out.println(line);}br.close();}}
newLine()与\r\n的区别
- newLine()是跨平台的方法
- \r\n只支持Windows系统
LineNumberReader
- LineNumberReader是BufferedReader的子类,具有相同的功能,并且可以统计行号
- 调用getLineNumber()方法可以获取当前行号
- 调用setLineNumber()方法可以设置当前行号
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.io.LineNumberReader;public class Demo5_LineNumberReader {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);//101:a 102:b 103:c 104:d 105:e}lnr.close();}}
装饰设计模式
一. 装饰设计模式的好处是
- 耦合性不强,被装饰类的变化与装饰类的变化无关
public class Demo6_Warp {public static void main(String[] args) {HeiMaStudent hs = new HeiMaStudent(new Student());//向装饰类中传入被装饰类hs.code();//javase javaweb ssh 数据库 大数据}}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("shh");System.out.println("数据库");System.out.println("大数据");}}
使用指定的码表读写字符
- FileReader是使用默认码表读取文件,如果需要使用指定码表读取,那么可以使用InputStreamReader(字节流对象,编码表):字节转字符
- FileWriter是使用默认码表写出文件,如果需要使用指定码表写出,那么可以使用InputStreamWriter(字节流对象,编码表):字符转字节
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.OutputStream;
import java.io.OutputStreamWriter;
import java.io.UnsupportedEncodingException;public class Demo7_TransIO {public static void main(String[] args) throws IOException {//demo1();//demo2();//demo3();}public static void demo3() throws UnsupportedEncodingException, FileNotFoundException, IOException {//使用BufferedReader,BufferedWriter包装类包装InputStreamReader,使其更强大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码表读(你好你好)FileWriter fw = new FileWriter("gbk.txt");//使用gbk码表写(浣犲ソ浣犲ソ)int c;while((c = fr.read()) != -1) {fw.write(c);}fr.close();fw.close();}}
包装类图解
递归
一.概述
- 方法自己调用自己
二.递归的利与弊
- 递归的弊端:不能调用次数太多,容易导致栈内存溢出
- 递归的好处:不用知道循环次数
三.构造方法不能使用递归调用
四.递归调用是否必须有返回值?
- 不一定(可以有,可以没有)
public class Demo8_Digui {public static void main(String[] args) {System.out.println(fun(6));}public static int fun(int n) {if(n == 1)return 1;elsereturn n * fun(n-1);}}
案例演示
获取某个文件夹下的所有.java文件,递归打印
import java.io.File;
import java.util.Scanner;public class Test5 {public static void main(String[] args) {File dir = getDir();printJavaFile(dir);}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;}}}public static void printJavaFile(File dir) {File[] subFiles = dir.listFiles(); //创建文件夹数组存储文件和文件夹for (File subFile : subFiles) {if(subFile.isFile() && subFile.getName().endsWith(".java")) {System.out.println(subFile);} else if(subFile.isDirectory()) {printJavaFile(subFile);//递归}}}}