博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
JAVA语言基础-面向对象(IO:IO字符流、递归)
阅读量:5952 次
发布时间:2019-06-19

本文共 17560 字,大约阅读时间需要 58 分钟。

hot3.png

21.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();                                             //关流

    

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.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();

21.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();

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
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 {    @Override    public void code() {        System.out.println("javase");        System.out.println("javaweb");    }}class HeiMaStudent implements Coder {    private Student s;                      //获取到被包装的类的引用    public HeiMaStudent(Student s) {        //通过构造函数创建对象的时候,传入被包装的对象        this.s = s;    }    @Override    public 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 {	@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(字节流,编码表)
  • 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
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,创建双列集合对象TreeMap		TreeMap
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从键盘读取一行

转载于:https://my.oschina.net/u/3668429/blog/1928665

你可能感兴趣的文章
【MSDN 目录】C#编程指南、C#教程、ASP.NET参考、ASP.NET 4、.NET Framework类库
查看>>
jquery 怎么触发select的change事件
查看>>
angularjs指令(二)
查看>>
<气场>读书笔记
查看>>
领域驱动设计,构建简单的新闻系统,20分钟够吗?
查看>>
web安全问题分析与防御总结
查看>>
React 组件通信之 React context
查看>>
Linux下通过配置Crontab实现进程守护
查看>>
ios 打包上传Appstore 时报的错误 90101 90149
查看>>
密码概述
查看>>
jQuery的技巧01
查看>>
基于泛型实现的ibatis通用分页查询
查看>>
gopacket 使用
查看>>
AlertDialog对话框
查看>>
我的友情链接
查看>>
linux安全---cacti+ntop监控
查看>>
鸟哥的linux私房菜-shell简单学习-1
查看>>
nagios配置监控的一些思路和工作流程
查看>>
通讯组基本管理任务三
查看>>
Centos下基于Hadoop安装Spark(分布式)
查看>>