java day21 IO流 字符流 FileReader FileWriter

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/amy260231120/article/details/83413937

(一)“字节”的定义

字节(Byte)是一种计量单位,表示数据量多少,它是计算机信息技术用于计量存储容量的一种计量单位。

(二)“字符”的定义

字符是指计算机中使用的文字和符号,比如1、2、3、A、B、C、~!·#¥%……—*()——+、等等。


21.01 字符流FileReader读

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

21.02 字符流FileWriter写

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

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

21.03 字符流的拷贝

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

21.04 什么情况下使用字符流

  • 字符流也可以拷贝文本文件, 但不推荐使用. 因为读取时会把字节转为字符, 写出时还要把字符转回字节.
  • 程序需要读取一段文本, 或者需要写出一段文本的时候可以使用字符流
  • 读取的时候是按照字符的大小读取的,不会出现半个中文
  • 写出的时候可以直接将字符串写出,不用转换为字节数组
    在这里插入图片描述

21.05 字符流不可以拷贝非纯文本的文件

  • 不可以拷贝非纯文本的文件
  • 因为在读的时候会将字节转换为字符,在转换过程中,可能找不到对应的字符,就会用?代替,写出的时候会将字符转换成字节写出去
  • 如果是?,直接写出,这样写出之后的文件就乱了,看不了了

21.06 自定义字符数组的拷贝

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

21.07 带缓冲的字符流

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

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

  • 缓冲区大小:1024*8
    BufferedReader br = new BufferedReader(new FileReader(“aaa.txt”)); //创建字符输入流对象,关联aaa.txt
    BufferedWriter bw = new BufferedWriter(new FileWriter(“bbb.txt”)); //创建字符输出流对象,关联bbb.txt

      int ch;				
      while((ch = br.read()) != -1) {		//read一次,会先将缓冲区读满,从缓冲去中一个一个的返给临时变量ch
      	bw.write(ch);					//write一次,是将数据装到字符数组,装满后再一起写出去
      }
      
      br.close();							//关流
      bw.close();  
    

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

21.09 将文本反转

  • 将一个文本文档上的文本反转,第一行和倒数第一行交换,第二行和倒数第二行交换
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 LineNumberReader类

  • LineNumberReader是BufferedReader的子类, 具有相同的功能, 并且可以统计行号
    • 调用getLineNumber()方法可以获取当前行号
    • 调用setLineNumber()方法可以设置当前行号
    • 有个成员变量叫lineNumber,readLine()读一次,就会自动➕一次
  •   LineNumberReader lnr = new LineNumberReader(new FileReader("aaa.txt"));
      String line;
      lnr.setLineNumber(100);									//设置行号,设置第一行为101
      while((line = lnr.readLine()) != null) {
      	System.out.println(lnr.getLineNumber() + ":" + line);//获取行号
      }
      
      lnr.close(); 
    

21.11 装饰设计模式

  • Student类和升级后的HeiMaStudent类耦合性并没有那么强,Student 类被改变后,HeiMaStudent类也不会变化。

  • 假如使用继承的话,耦合性太强了。

  • 总结:耦合性不强,被装饰类和装饰类的关系不强。

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

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

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

21.13 转换流图解

在这里插入图片描述

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

  • 获取一个文本上每个字符出现的次数,将结果写在times.txt上
	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 '\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流知识,模拟试用版软件,试用10次机会,执行一次就提示一次您还有几次机会,如果次数到了提示请购买正版
	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 + ""); // 必须转成String写入
			fw.close();
		}else {
			System.out.println("您的试用次数已到,请购买正版");
		}
		//关闭流
		br.close();
	}

21.16 (递归)

  • 5的阶乘
public static void main(String[] args) throws IOException {
		// TODO Auto-generated method stub
		digui(5);
		
	}
	public static void digui(int number) throws IOException {
		if (number == 1) {
			return;
		} 
		BufferedReader bufferedReader = new BufferedReader(new FileReader("zzz.txt"));
		Integer total = Integer.parseInt(bufferedReader.readLine());
		FileWriter fileWriter = new FileWriter("zzz.txt");
		fileWriter.write(total * number + "");
		bufferedReader.close();
		fileWriter.close();
		digui(--number);

	}

21.17 (练习)

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

猜你喜欢

转载自blog.csdn.net/amy260231120/article/details/83413937