版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
###21.01_IO流(字符流FileReader)
- 1.字符流是什么
- 字符流是可以直接读写字符的IO流
- 字符流读取字符, 就要先读取到字节数据, 然后转为字符. 如果要写出字符, 需要把字符转为字节再写出
- 2.FileReader
- FileReader类的read()方法可以按照字符大小读取
import java.io.FileReader;
import java.io.IOException;
public class Main {
public static void main(String[] args) throws IOException {
FileReader fr = new FileReader("xxx.txt");
int c; //一定要定义为int型因为FileReader返回类型为int型
if((c=fr.read())!=-1){
System.out.println((char)c); //强转为char显示
}
}
}
###21.02_IO流(字符流FileWriter)
-
FileWriter类的write()方法可以自动把字符转为字节写出
FileWriter fw = new FileWriter("aaa.txt"); fw.write("aaa"); 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();
//writer类中有一个2k的缓冲区,如果不关流就会将内容写到缓冲区,关流就会将缓冲区的内容刷新
###21.04_IO流(什么情况下使用字符流)
- 字符流也可以拷贝文本文件, 但不推荐使用. 因为读取时会把字节转为字符, 写出时还要把字符转回字节.
- 程序需要读取一段文本, 或者需要写出一段文本的时候可以使用字符流
- 只读或只写可以用字符流,其他情况用字节流好一点
- 读取的时候是按照字符的大小读取的,不会出现半个中文
- 写出的时候可以直接将字符串写出,不用转换为字节数组
###21.05_IO流(字符流是否可以拷贝非纯文本的文件)
- 不可以拷贝非纯文本的文件
- 因为在读的时候会将字节转换为字符,在转换过程中,可能找不到对应的字符,就会用?代替,写出的时候会将字符转换成字节写出去
- 如果是?,直接写出,这样写出之后的文件就乱了,看不了了
###21.06_IO流(自定义字符数组的拷贝)
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
public class Main {
public static void main(String[] args) throws IOException {
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_IO流(带缓冲的字符流)
- BufferedReader的read()方法读取字符时会一次读取若干字符到缓冲区, 然后逐个返回给程序, 降低读取文件的次数, 提高效率
- BufferedWriter的write()方法写出字符时会先写到缓冲区, 缓冲区写满时才会写到文件, 降低写文件的次数, 提高效率
import java.io.*;
public class Main {
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new FileReader("xxx.txt"));
//创建字符输入流对象,关联aaa.txt
BufferedWriter bw = new BufferedWriter(new FileWriter("yyy.txt"));
//创建字符输出流对象,关联bbb.txt
int len;
while((len = br.read()) != -1){
//read一次,会先将缓冲区读满,从缓冲去中一个一个的返给临时变量ch
bw.write(len);
//write一次,是将数据装到字符数组,装满后再一起写出去
}
br.close();//关流
bw.close();
}
}
###21.08_IO流(readLine()和newLine()方法)
- BufferedReader的readLine()方法可以读取一行字符(不包含换行符号)
- BufferedWriter的newLine()可以输出一个跨平台的换行符号"\r\n"
import java.io.*;
public class Main {
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));
String line;
while ((line = br.readLine()) != null){ //可以整行读取
System.out.println(line);
}
br.close();
}
}
import java.io.*;
public class Main {
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));
BufferedWriter bw = new BufferedWriter(new FileWriter("xxx.txt"));
String line;
while ((line = br.readLine()) != null){
bw.write(line);
bw.newLine(); //写出回车换行符
//bw.write("/r/n"); //换行符
//newLine是跨平台的方法,/r/n只支持windows系统
}
br.close();
bw.close();
}
}
###21.09_IO流(将文本反转)
- 将一个文本文档上的文本反转,第一行和倒数第一行交换,第二行和倒数第二行交换
import java.io.*;
import java.util.ArrayList;
public class Main {
/**
* @param args
* @throws IOException
* 1.创建输入输出流对象
* 2.创建集合对象,临时存储
* 3.将读到的数据存储在集合中
* 4.倒着遍历将数据写在文件上
* 5.关流
*
* 注意事项:流对象最好晚开早关
*/
public static void main(String[] args) throws IOException {
// BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));
// BufferedWriter bw = new BufferedWriter(new FileWriter("xxx.txt"));
ArrayList<String> list = new ArrayList<>();
BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));
BufferedWriter bw = new BufferedWriter(new FileWriter("xxx.txt"));
String line;
while ((line = br.readLine()) != null){
list.add(line);
}
for (int i = list.size(); i >= 0; i++) {
bw.write(list.get(i));
bw.newLine();
}
br.close();
bw.close();
}
}
###21.10_IO流(LineNumberReader)
- LineNumberReader是BufferedReader的子类, 具有相同的功能, 并且可以统计行号
- 调用getLineNumber()方法可以获取当前行号
- 调用setLineNumber()方法可以设置当前行号
import java.io.*;
public class Main {
public static void main(String[] args) throws IOException {
LineNumberReader lnr = new LineNumberReader(new FileReader("xxx.txt"));
String line;
lnr.setLineNumber(100); //设置初始值
while((line = lnr.readLine()) != null){
System.out.println(lnr.getLineNumber()+":"+line);
//从101开始,readLine一次,LineNumber就加一次
}
lnr.close();
}
}
###21.11_IO流(装饰设计模式)
- 装饰模式
- 1.获取被装饰类的引用
- 2.在构造方法中传入被装饰类的对象
- 3.对原有的功能进行升级
public class Main {
/**
* 装饰模式的好处:降低耦合性,被装饰的类的变化与装饰类的变化无关
*/
public static void main(String[] args){
CsStudent cs = new CsStudent(new Student1());
cs.code();
}
}
interface Coder{
public void code();
}
class Student1 implements Coder{
@Override
public void code() {
System.out.println("javase");
System.out.println("javaweb");
}
}
class CsStudent implements Coder{
//1.获取被装饰类的引用
private Student1 s; //获取学生引用
//2.在构造方法中传入被装饰类的对象
public CsStudent(Student1 s){
this.s = s;
}
//3.对原有的功能进行升级
@Override
public void code() {
s.code();
System.out.println("ssh");
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();
###21.13_IO流(转换流图解)
- 画图分析转换流
###21.14_IO流(获取文本上字符出现的次数)
- 获取一个文本上每个字符出现的次数,将结果写在times.txt上
import java.io.*;
import java.util.TreeMap;
public class Main {
/**
* 1.创建带缓冲的输入流对象
* 2.创建双列集合对象TreeMap
* 3.将读到的字符存储在双列集合中,存储时判断若不包含此键就将键和1存储,若不包含就将键和值加1存储
* 4.关闭输入流
* 5.创建输出流对象
* 6.遍历集合,将集合中的内容写到times.txt中
*/
public static void main (String[] args) throws IOException {
BufferedReader br = new BufferedReader(new FileReader("xxx.txt"));
TreeMap<Character,Integer> tm = new TreeMap<>();
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);
}
br.close();
BufferedWriter bw = new BufferedWriter(new FileWriter("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:
break;
}
bw.newLine();
}
bw.close();
}
}
###21.15_IO流(试用版软件)
- 当我们下载一个试用版软件,没有购买正版的时候,每执行一次就会提醒我们还有多少次使用机会用学过的IO流知识,模拟试用版软件,试用10次机会,执行一次就提示一次您还有几次机会,如果次数到了提示请购买正版
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
public class Main {
/**
* 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){
//在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的阶乘
public class Main {
/**
* 递归:方法自己调用自己
* 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) {
System.out.println(fun(5));
}
public static int fun(int num){
if(num == 1){
return 1;
}else{
return num * fun(num - 1);
}
}
}
- 递归的弊端:如果调用次数过多容易导致栈内存溢出
- 递归的好处:不用知道循环次数
- 构造方法是否可以递归调用:构造方法不能使用递归调用
- 递归调用是否必须有返回值:不一定,可以有也可以没有
###21.17_File类(练习)
- 需求:从键盘输入接收一个文件夹路径,打印出该文件夹下所有的.java文件名
import java.io.File;
import java.util.Scanner;
public class Main {
/**
* 从键盘接收一个文件夹路径
* 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);
}else if(subFile.isDirectory()){
//4.如果是文件夹,就递归调用
printJavaFile(subFile);
}
}
}
}
###21.18_IO流(总结)
- 1.会用BufferedReader读取GBK码表和UTF-8码表的字符
- 2.会用BufferedWriter写出字符到GBK码表和UTF-8码表的文件中
- 3.会使用BufferedReader从键盘读取一行