Day21
- IO流(字符流FileReader)
(1) 字符流是什么
字符流是可以直接读写字符的IO流
字符流读取字符,就要先读取到字节数据,然后转为字符。如果写出字符,需要把字符转为字节再写出。
(2) FileReader,FileWriter
FileReader类的read()方法可以按照大小读取
FileWriter类的writer()方法可以自动把字符转为字节写出
public class Demo1_FileReader {
public static void main(String[] args) throws IOException {
//demo1();
FileReader fr = new FileReader("xxx.txt");
int c;
while ((c = fr.read()) != -1) { //通过项目默认的码表,一次读取一个字符
char x = (char) c;
System.out.print(x);
}
}
private static void demo1() throws 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();
}
}
- IO流(字符流FileWriter)
FileWriter类的writer()方法可以自动把字符转为字节写出
public class Demo2_FileWriter {
public static void main(String[] args) throws IOException {
FileWriter fw = new FileWriter("yyy.txt");
fw.write("大家好,基础班快接近尾声了");
fw.close();
}
}
- IO流(字符流的拷贝)
public class Demo3 {
public static void main(String[] args) throws IOException {
FileReader fr = new FileReader("xxx.txt");
FileWriter fw = new FileWriter("copy.txt");
int c;
while ((c = fr.read()) != -1){
fw.write(c);
}
fr.close();
fw.close(); //Writer类中有一个2K的小缓冲区,如果不管流,就会将内容写到缓冲区中
//关流可以将缓冲区中内容刷新出来
}
}
- IO流(什么情况下使用字符流)
字符流也可以拷贝文本文件,但不推荐使用,因为读取时会把字节转为字符,写出时还要把字符转回字节。
程序需要读取一段文本,或者需要写出一段文本的时候可以使用字符流。
读取的时候是按照字符的大小读取的,不会出现半个中文;写出的时候可以直接按照字符串写出,不用转换成字节数组。
- IO流(字符流是否可以拷贝非纯文本的文件)
不可以拷贝非纯文本的文件。
因为在读的时候会将字节转换为字符,在转换过程中,可能找不到对应的字符,就会用?代替,写出的时候会将字符转换成字节写出去。
如果是?,直接写出,这样写出之后的文件就乱了,看不了了。 - IO流(自定义字符数组的拷贝)
public class Demo3 {
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(arr)) != -1){ //将文件上的数据读取到数组中
fw.write(arr,0,len); //将字符数组中的数据写到文件上
}
fr.close();
fw.close();
}
- IO流(带缓冲的字符流)
BufferedReader的read()方法读取字符时会一次读取若干字符到缓冲区,然后逐个返回给程序,降低读取文件的次数,提高效率。
BufferedWriter的Writer ()方法写出字符时会先写道缓冲区,缓冲区写满时才会写到文件,降低写文件的次数,提高效率。
public class Demo4 {
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();
}
}
- IO流(readLine()和newLine()方法)
BufferedReader的readLine ()方法可以读取一行字符(不包含换行符号)
BufferedWriter的newLine ()可以输出一个跨平台的换行符号”\r\n”
public class Demo5 {
/*
newLine()与\r\n的区别:
newLine()是跨平台的方法
\r\n支持的是windows系统
*/
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new FileReader("xxx.txt"));
BufferedWriter bw = new BufferedWriter(new FileWriter("aaa.txt"));
String line;
while ((line = br.readLine()) != null){
bw.write(line);
bw.newLine();
}
br.close();
bw.close();
}
private static void demo1() throws IOException {
BufferedReader br = new BufferedReader(new FileReader("xxx.txt"));
String line;
while ((line = br.readLine()) != null) {
System.out.println(line);
}
br.close();
}
}
- IO流(将文本反转)
将一个文本文档上的文本反转,第一行和倒数第一行交换,第二行和倒数第二行交换。
public class Test1 {
/*
将一个文本文档上的文本反转,第一行和倒数第一行交换,第二行和倒数第二行交换。
1:创建输入输出流
2:创建集合对象
3:将读到的数据写到文件中
4:倒着遍历集合中的数据写到文件上
5:关流
注意事项:
流对象尽可能晚开早关
*/
public static void main(String[] args) throws IOException {
// 1:创建输入输出流
BufferedReader br = new BufferedReader(new FileReader("xxx.txt"));
BufferedWriter bw = new BufferedWriter(new FileWriter("yyy.txt"));
//2:创建集合对象
ArrayList<String> list = new ArrayList<>();
// 3:将读到的数据写到文件中
String line;
while ((line = br.readLine()) != null){
list.add(line);
}
br.close(); //关流
//4:倒着遍历集合中的数据写到文件上
for(int i = list.size() - 1;i >= 0; i--){
bw.write(list.get(i));
bw.newLine();
}
bw.close();
}
}
- IO流(LineNumberReader)
LineNumberReader是BufferedReader的子类,具有相同的功能,并且可以统计行号。
调用getLineNumber()方法可以获取当前行号
调用setLineNumber()方法可以设置当前行号
public class Demo6_LineNumberReader {
/*
LineNumberReader是BufferedReader的子类,具有相同的功能,并且可以统计行号。
*/
public static void main(String[] args) throws IOException {
LineNumberReader lnr = new LineNumberReader(new FileReader("xxx.txt"));
String line;
lnr.setLineNumber(100); //设置100,就从101开始设置
while ((line = lnr.readLine()) != null) {
System.out.println(lnr.getLineNumber() + ":" + line);
}
lnr.close();
}
}
- IO流(装饰设计模式)
public class Demo7 {
/*
装饰设计模式的好处:
耦合性不强,被装饰类的变化与装饰类的变化无关
*/
public static void main(String[] args) {
HeiMaStudent hms = new HeiMaStudent(new Student() );
hms.code();
}
}
interface Coder {
public void code();
}
class Student implements Coder {
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:对原有的功能进行升级
public void code() {
s.code();
System.out.println("ssh");
System.out.println("数据库");
System.out.println("大数据");
System.out.println("...");
}
}
- IO流(使用指定的码表读写字符)
FileReader是使用默认码表读取文件,如果需要使用指定码表读取,那么可以使用InputStreamReader(字节流,编码表)
FileWriter是使用默认码表写出文件,如果需要使用指定码表写出,那么可以使用OutputStreamWriter (字节流,编码表)
public class Demo8_TransIO {
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream("utf_8.txt"),"utf-8");
//更高效读
BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(new FilterOutputStream("mmm.txt"),"gbk"))
//更高效写
int c;
while ((c = br.read()) != -1){
bw.write(c);
}
br.close();
bw.close();
}
private static void demo2() throws IOException {
InputStreamReader isr = new InputStreamReader(new FileInputStream("utf_8.txt"),"uTf-8");
//指定码表读字符
OutputStreamWriter osw = new OutputStreamWriter(new FilterOutputStream("aaa.txt"),"gbk");
//指定码表写字符
int c;
while ((c = isr.read()) != -1){
osw.write(c);
}
isr.close();
osw.close();
}
private static void demo1() throws IOException {
//用默认编码表读写,出现乱码
FileReader fr = new FileReader("utf_8.txt");
FileWriter fw = new FileWriter("mmm.txt");
int b;
while ((b = fr.read()) != -1){
fw.write(b);
}
fr.close();
fw.close();
}
}
- IO流(转换流图解)
画图分析转换流
- IO流(获取文本上字符出现的次数)
获取一个文本上每个字符出现的次数,将结果写在times.txt上
public class Test2 {
/*
获取一个文本上每个字符出现的次数,将结果写在times.txt上
1:创建带缓冲的输入流对象
2:创建双列集合对象TreeMap
3:将读到的字符存储在双列集合中,存储要做判断,如果不包含这个键就将键和1存储,就将键和值加1存储
4:关闭输入流
5:创建输出流对象
6:遍历集合,将集合中的内容写到times.txt中
7:关闭输出流
*/
public static void main(String[] args) throws IOException {
//1:创建带缓冲的输入流对象
BufferedReader br = new BufferedReader(new FileReader("aaa.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();
}
}
- IO流(试用版软件)
当我们下载一个试用版软件,没有购买到正版是,每执行一次就会提醒我们还有多少次使用机会用学过的IO流知识,模拟试用版软件,试用10次机会,执行一次就提示一次您还有几次机会,如果次数到了提示请购买正版。
public class Test3 {
/*
当我们下载一个试用版软件,没有购买到正版是,每执行一次就会提醒我们还有多少次使用机会.
用学过的IO流知识,模拟试用版软件,试用10次机会,执行一次就提示一次您还有几次机会,
如果次数到了提示请购买正版。
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) {
System.out.println("您还有" + times-- + "次机会");
FileWriter fw = new FileWriter("config.txt");
fw.write(times + "");
fw.close();
}else {
System.out.println("您的试用次数已到,请购买正版");
}
br.close();
}
}
- File类(递归)
5的阶乘
递归的弊端:不能调用次数过多,会造成栈溢出
递归的好处:不用知道循环次数
构造方法是否可以进行递归调用?
构造方法不可以递归调用
递归调用是否必须有返回值?
不一定(可以有,也可以没有)
public class Demo9_Digui {
/*
递归:自己调用自己
求: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);
}
}
//用循环做
private static void demo1() {
int reslut = 1;
for(int i =1; i <= 5; i++){
reslut = reslut * i;
}
System.out.println(reslut);
}
}
- File类(练习)
从键盘输入接收一个文件加路径,打印出该文件夹下所有的.java文件名
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:返回值类型File
2:不需要有参数
*/
public static void printJavaFile(File dir){
//获取到该文件夹路径下的所有文件和文件夹,存储在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);
}
}
}
}
- IO流(总结)
(1) 会用BufferedReader读取GBK码表和UTF-8码表的字符
(2) 会用BufferedWriter写出字符到GBK码表和UTF-8码表的文件中
(3) 会使用BufferedReader从键盘读取一行。