文章目录
- 1.File类
- 2.字节流
- 3.字符流
-
-
- 3.1 为什么出现字符流?
- 3.2 编码表
- 3.3 字符串中的编码解码问题
- 3.4 字符流中的编码解码问题
- 3.5 字符流写数据的5种方式
- 3.6字符流读数据的两种方式
- 3.7 案例 复制java文件(改进版)
- 3.8 字符缓冲流
- 3.9 案例:复制java文件(字符缓冲流改进版)
- 3.10 字符缓冲流特有功能
- 3.11 复制java文件(字符缓冲流特有功能改进版)
- 3.12 IO流总结
- 3.13 案例:集合到文件
- 3.14 文件到集合
- 3.15点名器
- 3.16 集合到文件(改进版)
- 3.17 文件到集合(改进版)
- 3.18集合到文件(数据排序改进版)
- 3.19 案例:复制单级文件夹
- 3.20 案例:复制多级文件夹
- 3.21复制文件的异常处理
-
- 4.特殊操作流
1.File类
1.1File类概述和构造方法
File:它是文件和目录路径敌人抽象表示
- 文件和目录是可以通过File封装成对象的
- 对于File而言,其封装的并不是一个真正存在的文件,仅仅是一个个路径而已。它可以是存在的,也可以是不存在的。将来是要通过具体的操作把这个路径的内容转换为具体存在的。
方法名 | 说明 |
---|---|
File(String pathname) | 通过给定的路径名字符串转换为抽象路径名来创建新的File实例 |
File(String parent,String child) | 从父路径名字符串和子路径名字符串创建新的File实例 |
File(File parent,String child) | 从父抽象路径名和子路径名字符串创建新的File实例 |
1.2 File类创建功能
方法名 | 说明 |
---|---|
public boolean creatNewFile() | 当具有该名称的文件不存在时,创建一个由该抽象路径名命名的新空文件 |
public boolean mkdir() | 创建由此抽象路径命名的目录 |
public boolean mkdirs() | 创建由此抽象路径命名的目录,包括任何必需但不存在的父目录,多级目录 |
1.3Fil类删除功能
方法名 | 说明 |
---|---|
public boolean delete() | 删除由此抽象路径名表示的文件或目录 |
绝对路径和相对路径的区别
- 绝对路径:完整的路径名,不需要任何其他信息就可以定位到它表示的文件,例如: E:\FileTest\java.txt
- 相对路径:必须使用取自其他路径名的信息进行注释。例如:myFile\java.txt
删除目录时的注意事项:
- 如果一个目录中有内容(目录,文件),不能直接删除。应该先删除目录中的内容,最后才能删除目录
1.4File类判断和获取功能
方法名 | 说明 |
---|---|
public boolean isDirectory() | 测试此抽象路径名表示的File是否为目录 |
public boolean isFile() | 测试此抽象路径名表示的File是否为文件 |
public boolean exists() | 测试此抽象路径名表示的File是否存在 |
public String getAbsolutePath() | 返回此抽象路径名的绝对路径名字符串 |
public String getPath() | 将此抽象路径名转换为路径名字符串 |
public String getName() | 返回此抽象路径名表示的文件和目录的名称 |
public String[] list() | 返回此抽象路径名表示的目录中的文件和目录的名称字符串数组 |
public File[] listFiles() | 返回此抽象路径名表示的路径中的文件和目录的File对象数组 |
package com.fun01;
import java.io.File;
public class FileDemo04 {
public static void main(String[] args) {
File f1 = new File("G:\\FileTest");
// System.out.println(f1.isDirectory());
// System.out.println(f1.isFile());
// System.out.println(f1.exists());
// System.out.println(f1.getAbsolutePath());
// System.out.println(f1.getPath());
// System.out.println(f1.getName());
/* String[] list = f1.list();
for(String s : list){
System.out.println(s);
}*/
File[] files = f1.listFiles();
for(File f : files){
System.out.println(f);
}
1.5递归
递归概述:以编程的角度来看,递归指的是方法定义中调用方法本身的现象
递归解决问题的思路:
把一个复杂的问题转化为一个与原问题相似的规模较小的问题来求解
递归策略只需少量的程序就可描述出解题过程所需要的多次重复计算
递归解决问题要找到两个内容:
- 递归出口:否则会出现内存溢出
- 递归规则:与原问题相似的规模较小问题
1.5递归求阶乘
需求:用递归求5的阶乘
思路:
①定义一个方法,用于递归阶乘,参数为一个int类型的变量
②在方法内部判断该变量的值是否等于1
是:返回1
否:返回n*(n-1)!
③调用方法
④输出结果
public class DiGuiDemo02 {
public static void main(String[] args){
System.out.println("5的阶乘是:" + jc(5));
}
public static int jc(int n){
if(n == 1){
return 1;
}else{
return n * jc(n-1);
}
}
}
1.6 遍历目录
需求:给定一个路径(G:\\FileTset),请通过地柜完成遍历该目录下的所有内容,并把所有文件的绝对路径输出在控制台
思路:
①根据给定的路径创建一个File对象
②定义一个方法,用于获取给定目录下的所有内容,参数为第一步创建的File对象
③获取给定的File目录下的所有文件或者目录的File数组
④遍历File数组,得到每一个File对象
⑤判读该File对象是否是目录
是:递归调用
否:获取据对路径输出在控制台
⑥调用方法
package com.fun01;
import java.io.File;
/**
* @author 23389.
* @date 2021/7/3.
*/
/*
需求:给定一个路径(G:\\\FileTset),请通过地柜完成遍历该目录下的所有内容,并把所有文件的绝对路径输出在控制台
思路:
①根据给定的路径创建一个File对象
②定义一个方法,用于获取给定目录下的所有内容,参数为第一步创建的File对象
③获取给定的File目录下的所有文件或者目录的File数组
④遍历File数组,得到每一个File对象
⑤判读该File对象是否是目录
是:递归调用
否:获取据对路径输出在控制台
⑥调用方法
*/
public class DiGuiDemo03 {
public static void main(String[] args){
File file = new File("G:\\FileTest");
getAllFilePath(file);
}
public static void getAllFilePath(File file){
//获取给定的File目录下的所有文件或者目录的File数组
File[] files = file.listFiles();
// 遍历File数组,得到每一个File对象
if(files != null){
for(File file1 : files){
//判断是否为文件
if(file1.isFile()){
System.out.println(file1.getAbsolutePath());
}else{
getAllFilePath(file1);
}
}
}
}
}
2.字节流
2.1 IO流概述和分类
IO流分类:
- 按照数据的流向
输入流:读数据
输出流:写数据
-
按照数据类型来划分
字节流
字节输入流;字节输出流
字符流
字符输入流;字符输出流
一般来说,我们说的IO流的分类是按照数据类型来划分的
那么这两种流都在什么情况下使用呢?
-
如果数据通过Windows自带的记事本软件打开,我们还可以读懂里面的内容,就使用字符流
否则使用字节流。如果你不知道使用哪种软件打开,就使用字节流。
2.2 字节流读写数据
字节流抽象基类
- InputStream:这个抽象类是表示字节输入流的所有类的超类
- OutStream:这个抽象类是表示字节输出流的所有类的超类
- 子类名特点:子类名称都是以其父类名作为子类名的后缀
FileOutputStream:文件输出流,用于将数据写入文件
- FileOutputStream(String name):创建文件输出流以指定的名称写入文件
使用字节输出流写数据的步骤
- 创建字节输出流对象(调用系统功能创建了文件,创建了字节输出流对象,让字节输出流对象指向文件)
- 调用字节输出流对象的写数据方法
- 释放资源(关闭此文件输出流并释放相关联的任何系统资源)
2.3字节流写数据的3种方式
方法名 | 说明 |
---|---|
void write(int b) | 将指定的字节写入此文件输出流,一次写一个字节数据 |
void write(byte[] b) | 将b.length字节从指定的字节数组写入此文件输出流,一次写一个字节数组数据 |
void write(byte[] b,int off ,int len) | 将len字节从指定的字节数组开始,从偏移量off开始写入此文件输出流,一次写一个字节数组的部分数据 |
2.4 字节流写数据的两个小问题
字节流写数据如何实现换行呢?
- 写完数据后,加换行符
windows:\r\n
linux:\n
mac:\r
字节流写数据如何实现追加写入呢?
- publicFileOutputStream(String name,boolean append)
- 创建文件输出流以指定的名称写入文件,如果第二个参数为true,则字节将写入文件的末尾而不是开头
2.5 字节流数据加异常处理
finally: 在异常处理时提供finally块来执行所有清除操作,比如IO流中的释放资源
特点:被finally控制的语句一定会执行,除非JVM退出
try{
可能出现异常的代码;
}catch{
异常的处理代码;
}finally{
执行所有清除操作;
}
2.6 字节流读数据(一次读一个字节数组数据)
需求:把文件fos.txt的内容取出来在控制台输出
使用字节输入流读数据的步骤:
①创建字节输入流对象
②调用字节输入流对象的读数据方法
③释放资源
package com.fun01;
import java.io.FileInputStream;
import java.io.IOException;
/**
* @author 23389.
* @date 2021/7/10.
*/
public class ReadTxt {
public static void main(String[] args) throws IOException {
FileInputStream fis = new FileInputStream("myByteStream\\fos.txt");
int len;
byte[] bys = new byte[1024];
while((len = fis.read(bys)) != -1){
System.out.println(new String(bys,0,len));
}
fis.close();
}
}
2.7 复制图片
package com.fun01;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
/**
* @author 23389.
* @date 2021/7/4.
*/
public class CopyImg {
public static void main(String[] args) throws IOException {
//创建文件输入流对象
FileInputStream fis = new FileInputStream("G:\\FileTest\\icon.jpg");
//创建文件输出流对象
FileOutputStream fos = new FileOutputStream("myByteStream\\icon.jpg");
byte[] by = new byte[1024];
int len;
while((len = fis.read(by)) != -1){
fos.write(by,0,len);
}
//释放资源
fis.close();
fos.close();
}
}
2.8字节缓冲流
程序从输入流读取数据,向输出流写入数据
字节缓冲流:
- BufferedOutputStream:该类实现缓冲输出流,通过设置这样的输出流,应用程序可以向底层输出流写入字节,而不必为写入的每个字节导致底层系统的调用。
- BufferInputStream:创建BufferedInputStream将创建一个内部缓冲区数组。当从流中读取或跳过字节时,内部缓冲区将根据需要从所包含的输入流重新填充,一次很多字节。
构造方法:
- 字节缓冲输出流:BufferedOutputStream(OutputStream out)
- 字节缓冲输入流:BufferedInputStream(InputStream in)
为什么构造方法需要的时字节流,而不是具体的文件或路径呢?
- 字节缓冲流仅仅提供缓冲区,而真正的读写数据还得依靠基本的字节流对象进行操作。
2.9复制视频
需求:把"G:\\FileTest\\计科03班王重重.mkv"复制到模块目录下的"计科03班王重重.mkv"
思路:
①根据数据源创建字节输入流对象
②根据目的地创建字节输出流对象
③读写数据,复制视频
④释放资源
3.字符流
3.1 为什么出现字符流?
由于字节流操作中文不是特别的方便,所以Java就提供字符流
- 字符流 = 字节流 + 编码表
用字节流复制文本文件时,文本文件中也会有中文,但是没有问题,原因是最终底操作会自动进行字节拼接成中文,如何识别是中文呢?
- 汉字在存储的时候,无论选择哪种编码存储,第一个字节都是负数
3.2 编码表
基础知识:
-
计算机中储存的信息都是二进制数表示的;我们在屏幕上看到的英文,汉字等字符是二进制数转换之后的结果
-
按照某种规则,将字符存储在计算机中,称之为编码,反之,将存储在计算机中的二进制数按照某种规则解析显示出来,称为解码,这里强调一下:按照A编码存储,必须按照A编码解析,这样才能显示正确的文本符号。否则就会导致乱码现象。
字符编码:就是一套自然语言的字符与二进制数之间对应的规则(A 65)
字符集:
-
是一个系统所支持的所有字符的集合,包括各国家文字、标点符号、数字等
-
计算机要准确的存储和识别各种字符集符号,就需要进行字符编码,一套字符集至少有一套字符编码。
常见的字符集有ASCII编码,GBXXX字符集。Unicode字符集等。
ASCII字符集:
- ASCII:是基于拉丁字母的一套电脑编码系统,用于显示现代英语,主要包括控制字符(回车键、退格、换行键等)和可显示字符(英文大小写字符、阿拉伯数字和西文符号)
- 基本的ASCII字符集,使用7位表示一个字符,共128字符。ASCII的扩展字符集使用8位表示一个字符,共256字符,方便支持欧洲常用字符。是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数组等。
GBXXX字符集:
- GB2312:简体中文码表,一个小于127的字符的意义与原来相同,但大于127的字符连接在一起时,就表示一个汉字,这样大约可以组合了包含7000多个简体汉字,此外,数学符号、罗马希腊的字母、日文的假名等都编进去了,连在ASCII里本来就有的数字、标点、字母都统统重新编了两个字长的编码,这就是常说的”全角“字符,而原来在127号以下的那些就叫"半角"字符。
- GBK:最常用的中文码表,是在GB2312标准基础上的扩展规范,使用了双字节编码方案,共收录了21003个汉字,完全兼容GB2标准,同时支持繁体汉字以及日韩文等。
- GB8030:最新的中文码表。收录汉字70244个,采用多字节编码,每个字可以由1个、2个或4个字节组成。支持中国国内少数民族的文字,同时支持繁体汉字以及日韩汉文等。
Unicode字符集:
-
为表达任意语言的任意字符而设计,是业界的一种标准,也称为统一码、标准万国码。它最多使用4个字节的数字来表达每个字母、符号,或者文字。有三种编码方案,UTF-8、UTF-16和UTF-32.最常用的为UTF-8编码。
-
UTF-8编码‘可以用来表示Unicode标准中任意字符,它是电子邮件、网页及其他存储或传送文字的应用中,优先采用的编码。互联网工程工作小组(IETF)要求所有互联网协议都必须支持UTF-8编码。它使用一至四个字节为每个字符编码。
编码规则:
128个US-ASCII字符,只需要一个字节编码
拉丁文等字符,需要二个字节编码
大部分常用字(含中文),使用三个字节编码
其他极少使用的Unicode辅助字符,使用四字节编码
3.3 字符串中的编码解码问题
编码:
- byte[] getBytes():使用平台的默认字符集将该String编码为一系列字节,将结果存储到新的字节数组中
- byte[] getBytes(String charset(Name)):使用指定的字符集将该String编码为一系列字节,将结果存储到新的字节数组中。
解码:
- String(byte[] bytes): 通过使用平台的默认字符集解码指定的字节数组来构造新的String
- String(byte[] bytes,String charset(Name)):通过指定的字符集解码指定的字节数组来构造新的String
3.4 字符流中的编码解码问题
字符流抽象基类:
- Reader:字符输入流的抽象类
- Writer:字符输出流的抽象类
字符流中和编码解码问题相关的两个类:
- InputStreamReader
- OutputStreamWriter
3.5 字符流写数据的5种方式
方法名 | 说明 |
---|---|
void write(int c) | 写一个字符 |
void write(char[] cbuf) | 写入一个字符数组 |
void write(char[] cbuf,int off,int len) | 写入一个字符数组的一部分 |
void write(String str) | 写入一个字符串 |
void write(String str,int off,int len) | 写入一个字符串的一部分 |
3.6字符流读数据的两种方式
方法名 | 说明 |
---|---|
int read() | 一次读一个字符数据 |
int read(char[] cbuf) | 一次读一个字符数组数据 |
3.7 案例 复制java文件(改进版)
需求:把模块目录下的"ConversionStreamDemo.java",复制到模块目录下的"Copy.java"
分析:
①移除流的名字比较长,而我们常见的操作都是按照本地默认的编码实现的,所以,为了简化书写,转化流提供对应的子类
②FileReader:用于读取字符文件的便捷类
FileReader(String fileName)
③FileWriter:用于写入字符文件的便捷类
FileWriter(String fileName)
④数据源和目的地的分析:
数据源:myCharStream\\ConversionStreamDemo.java----读数据-----Reader-----InputStreamReader-----FileReader
目的地:myCharStream\\Copy.java----写数据-----Writer-----OutputStreamWriter------FileWriter
package com.fun03;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
/**
* @author 23389.
* @date 2021/7/7.
*/
/*
需求:
把模块目录下的ConversionStreamDemo.java复制到
模块目录下的Copy.java
思路:
1.根据数据源创建字符输入流对象
2.根据数据目的创建字符输出流对象
3.读写数据,复制文件
4.释放资源
*/
public class CopyJavaDemo01 {
public static void main(String[] args) throws IOException {
//根据数据源创建字符输入流对象
FileReader fr = new FileReader("myCharStream\\ConversionStreamDemo.java");
//根据数据目的创建字符输出流对象
FileWriter fw = new FileWriter("myCharStream\\Copy.java");
//读写数据,复制文件
//一次读写一个字符数据
// int ch;
// while((ch = fr.read()) != -1){
// fw.write(ch);
// }
//一次读写一个字符数组数据
char[] chs = new char[1024];
int ch;
while((ch = fr.read(chs)) != -1){
fw.write(chs,0,ch);
}
//释放资源
fr.close();
fw.close();
}
}
3.8 字符缓冲流
字符缓冲流:
- BufferedWriter:将文本写入字符输出流,缓冲字符,以提供单个字符,数组和字符串的高效写入,可以指定缓冲区大小,或者可以接收默认大小,或者可以接收默认大小,默认值是足够大。可用于大多数用途
- BufferedReader:从字符输入流读取文本,缓冲字符,以提供字符,数组个字符串的高效读取,可以指定缓冲区大小,或者可以使用默认大小,默认值足够大,可用于大多数用途。
构造方法:
- BufferedWriter(Writer out)
- BufferedReader(Reader in)
package com.fun04;
import java.io.*;
/**
* @author 23389.
* @date 2021/7/7.
*/
public class BufferedStreamDemo01 {
public static void main(String[] args) throws IOException {
//BufferedWriter(Writer out)
// FileWriter fw = new FileWriter("myCharStream\\bw.txt");
// BufferedWriter bw = new BufferedWriter(fw);
// BufferedWriter bw = new BufferedWriter(new FileWriter("myCharStream\\bw.txt"));
// bw.write("hello\r\n");
// bw.write("world\r\n");
// bw.close();
BufferedReader bw = new BufferedReader(new FileReader("myCharStream\\bw.txt"));
//一次读取一个字符数据
// int ch;
// while((ch = bw.read()) != -1){
// System.out.print((char) ch);
// }
//一次读取一个字符数组数据
int len;
char[] chs = new char[1024];
while((len = bw.read(chs)) != -1){
System.out.println(new String(chs