IO文件流
File类&递归思想
File类
概述和构造方法
介绍
- 它是文件和目录路径名的抽象表示
- 文件和目录是可以通过File封装成对象的
- 对于File而言,其封装的并不是一个真正存在的文件,仅仅是一个路径名而已。它可以是存在的,也可以是不存在的。将来是要通过具体的操作把这个路径的内容转换为具体存在的
构造方法
File(String pathname) :通过将给定的路径名字符串转换为抽象路径名来创建新的 File实例
File(String parent, String child) :从父路径名字符串和子路径名字符串创建新的 File实例
File(File parent, String child) :从父抽象路径名和子路径名字符串创建新的 File实例
创建功能
方法分类
public boolean createNewFile():当具有该名称的文件不存在时,创建一个由该抽象路径名命名的新空文件
public boolean mkdir(): 创建由此抽象路径名命名的目录
public boolean mkdirs():创建由此抽象路径名命名的目录,包括任何必需但不存在的父目录
判断和获取功能
判断功能
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对象数组
删除功能
方法分类
public boolean delete(): 删除由此抽象路径名表示的文件或目录
递归思想
递归
介绍
- 以编程的角度来看,递归指的是方法定义中调用方法本身的现象
- 把一个复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解
- 递归策略只需少量的程序就可描述出解题过程所需要的多次重复计算
基本使用
例子:不死神兔问题
注意事项
- 递归一定要有出口。否则内存溢出
- 递归虽然有出口,但是递归的次数也不宜过多。否则内存溢出
关于File类和递归的相关示例:
操作D盘下的my.txt文件
1) 判断my.txt文件是否存在
2) 若存在则删除;若不存在则创建
import java.io.File;
import java.io.IOException;
/*
操作D盘下的my.txt文件
1) 判断my.txt文件是否存在
2) 若存在则删除;若不存在则创建
*/
public class Test {
public static void main(String[] args) throws IOException {
// public boolean exists()测试此抽象路径名表示的File是否存在
//创建一个File对象
File f = new File("D:\\my.txt");
/**
* exists():判断文件是否存在
* delete():删除文件
* createNewFile():创建文件
*/
if (f.exists()) {
f.delete();
} else {
f.createNewFile();
}
}
}
字节流
IO流
IO流概述和分类
介绍
- IO:输入/输出(Input/Output)
- 流:是一种抽象概念,是对数据传输的总称。也就是说数据在设备间的传输称为流,流的本质是数据传输
- IO流就是用来处理设备间数据传输问题的。常见的应用:文件复制;文件上传;文件下载
分类
- 按照数据的流向
- 输入流:读数据
- 输出流:写数据
- 按照数据类型来分
- 字节流
- 字节输入流
- 字节输出流
- 字符流
- 字符输入流
- 字符输出流
使用场景
- 如果操作的是纯文本文件,优先使用字符流
- 如果操作的是图片、视频、音频等二进制文件。优先使用字节流
- 如果不确定文件类型,优先使用字节流。字节流是万能的流
字节流写数据
抽象基类
- InputStream:这个抽象类是表示字节输入流的所有类的超类
- OutputStream:这个抽象类是表示字节输出流的所有类的超类
- 子类名特点:子类名称都是以其父类名作为子类名的后缀
字节输出流
FileOutputStream(String name):创建文件输出流以指定的名称写入文件
步骤
- 创建字节输出流对象(调用系统功能创建了文件,创建字节输出流对象,让字节输出流对象指向文件)
- 调用字节输出流对象的写数据方法
- 释放资源(关闭此文件输出流并释放与此流相关联的任何系统资源)
字节流写数据的三种方式
数据的方法分类
void write(int b):将指定的字节写入此文件输出流 一次写一个字节数据
void write(byte[] b) : 将 b.length字节从指定的字节数组写入此文件输出流 一次写一个字节数组数据
void write(byte[] b, int off, int len):将 len字节从指定的字节数组开始,从偏移量off开始写入此文件输出流 一次写一个字节数组的部分数据
字节流写数据的两个小问题
字节流写数据如何实现换行
- windows:\r\n
- linux:\n
- mac:\r
字节流写数据如何实现追加写入
- public FileOutputStream(String name,boolean append)
- 创建文件输出流以指定的名称写入文件。如果第二个参数为true ,则字节将写入文件的末尾而不是开头
字节流写数据加异常处理
异常处理格式
// try-catch-finally
try{
可能出现异常的代码;
}catch(异常类名 变量名){
异常的处理代码;
}finally{
执行所有清除操作;
}
finally特点
被finally控制的语句一定会执行,除非JVM退出
字节流读数据(一次读一个字节数据)
字节输入流
FileInputStream(String name):通过打开与实际文件的连接来创建一个FileInputStream ,该文件由文件系统中的路径名name命名
字节输入流读取数据的步骤
- 创建字节输入流对象
- 调用字节输入流对象的读数据方法
- 释放资源
字节流读数据(一次读一个字节数组数据)
一次读一个字节数组的方法
- public int read(byte[] b):从输入流读取最多b.length个字节的数据
- 返回的是读入缓冲区的总字节数,也就是实际的读取字节个数
字节缓冲流
介绍
- BufferOutputStream:该类实现缓冲输出流。 通过设置这样的输出流,应用程序可以向底层输出流写入字节,而不必为写入的每个字节导致底层系统的调用
- BufferedInputStream:创建BufferedInputStream将创建一个内部缓冲区数组。 当从流中读取或跳过字节时,内部缓冲区将根据需要从所包含的输入流中重新填充,一次很多字节
构造方法
BufferedOutputStream(OutputStream out):创建字节缓冲输出流对象
BufferedInputStream(InputStream in):创建字节缓冲输入流对象
相关示例:
编写程序向文本文件中写入自己的信息,格式为:姓名:XXX 性别:X 年龄:XX 班级:XXX,将该信息读出后显示的屏幕上后把文件删除。
public class Test {
public static void main(String[] args) throws IOException {
Scanner sc = new Scanner(System.in);
//编写程序向文本文件中写入自己的信息,
File file = new File(
"E:\\IdeaProjects\\JavaSE_Code\\idea_test[关卡8--16]\\src\\work14\\bytestream\\lianxi7\\message.txt");
FileOutputStream fos = new FileOutputStream(
"E:\\IdeaProjects\\JavaSE_Code\\idea_test[关卡8--16]\\src\\work14\\bytestream\\lianxi7\\message.txt");
FileInputStream fis = new FileInputStream(
"E:\\IdeaProjects\\JavaSE_Code\\idea_test[关卡8--16]\\src\\work14\\bytestream\\lianxi7\\message.txt");
System.out.println("请输入自己的姓名:");
String name = sc.next();
name = "姓名:"+name;
System.out.println("请输入自己的年龄:");
String age = sc.next();
age = "年龄:"+age;
System.out.println("请输入自己的性别:");
String sex = sc.next();
sex = "性别:"+sex;
System.out.println("请输入自己的班级:");
String garde = sc.next();
garde = "班级:"+garde;
//格式为:姓名:XXX 性别:X 年龄:XX 班级:XXX,
fos.write(name.getBytes());
fos.write(" ".getBytes());
fos.write(age.getBytes());
fos.write(" ".getBytes());
fos.write(sex.getBytes());
fos.write(" ".getBytes());
fos.write(garde.getBytes());
fos.write(" ".getBytes());
System.out.println("成功!");
//将该信息读出后显示的屏幕上后把文件删除。
//将数据按一组一组数组读
byte[] bytes = new byte[2048];
//一个数组读取的实际字符个数
int len;
while ((len=fis.read(bytes)) != -1){
System.out.print(new String(bytes,0,len));
}
//释放资源
fos.close();
fis.close();
//必须在释放后删除!
file.delete();
}
}
字符流
字符流
字符流概述
介绍
由于字节流操作中文不是特别的方便,所以Java就提供字符流
字符流 = 字节流 + 编码表
中文的字节存储方式
用字节流复制文本文件时,文本文件也会有中文,但是没有问题,原因是最终底层操作会自动进行字节拼接成中文,如何识别是中文的呢?
汉字在存储的时候,无论选择哪种编码存储,第一个字节都是负数
编码表
字符集
是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等
l计算机要准确的存储和识别各种字符集符号,就需要进行字符编码,一套字符集必然至少有一套字符编码。常见字符集有ASCII字符集、GBXXX字符集、Unicode字符集等
常见字符集
ASCII字符集:
lASCII:是基于拉丁字母的一套电脑编码系统,用于显示现代英语,主要包括控制字符(回车键、退格、换行键等)和可显示字符(英文大小写字符、阿拉伯数字和西文符号)
基本的ASCII字符集,使用7位表示一个字符,共128字符。ASCII的扩展字符集使用8位表示一个字符,共256字符,方便支持欧洲常用字符。是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等
GBXXX字符集:
GBK:最常用的中文码表。是在GB2312标准基础上的扩展规范,使用了双字节编码方案,共收录了21003个汉字,完全兼容GB2312标准,同时支持繁体汉字以及日韩汉字等
Unicode字符集:
UTF-8编码:可以用来表示Unicode标准中任意字符,它是电子邮件、网页及其他存储或传送文字的应用 中,优先采用的编码。互联网工程工作小组(IETF)要求所有互联网协议都必须支持UTF-8编码。它使用一至四个字节为每个字符编码
编码规则:
128个US-ASCII字符,只需一个字节编码
拉丁文等字符,需要二个字节编码
大部分常用字(含中文),使用三个字节编码
其他极少使用的Unicode辅助字符,使用四字节编码
字符串中的编码解码问题
相关方法
byte[] getBytes():使用平台的默认字符集将该 String编码为一系列字节
byte[] getBytes(String charsetName):使用指定的字符集将该 String编码为一系列字节
String(byte[] bytes):使用平台的默认字符集解码指定的字节数组来创建字符串
String(byte[] bytes, String charsetName):通过指定的字符集解码指定的字节数组来创建字符串
字符流中的编码解码问题
字符流中和编码解码问题相关的两个类
它读取字节,并使用指定的编码将其解码为字符
它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集
OutputStreamWriter:是从字符流到字节流的桥梁
是从字符流到字节流的桥梁,使用指定的编码将写入的字符编码为字节
它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集
构造方法
InputStreamReader(InputStream in):使用默认字符编码创建InputStreamReader对象
InputStreamReader(InputStream in,String chatset):使用指定的字符编码创建InputStreamReader对象
OutputStreamWriter(OutputStream out):使用默认字符编码创建OutputStreamWriter对象
OutputStreamWriter(OutputStream out,String charset):使用指定的字符编码创建OutputStreamWriter对象
字符流写数据的五种方式
方法介绍
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) : 写一个字符串的一部分
刷新和关闭的方法
flush():刷新流,之后还可以继续写数据
close(): 关闭流,释放资源,但是在关闭之前会先刷新流。一旦关闭,就不能再写数据
字符流读数据的两种方式
方法介绍
int read() : 一次读一个字符数据
int read(char[] cbuf):一次读一个字符数组数据
字符缓冲流
字符缓冲流
介绍
BufferedWriter:将文本写入字符输出流,缓冲字符,以提供单个字符,数组和字符串的高效写入,可以指定缓冲区大小,或者可以接受默认大小。默认值足够大,可用于大多数用途
BufferedReader:从字符输入流读取文本,缓冲字符,以提供字符,数组和行的高效读取,可以指定缓冲区大小,或者可以使用默认大小。 默认值足够大,可用于大多数用途
构造方法
BufferedWriter(Writer out):创建字符缓冲输出流对象
BufferedReader(Reader in) :创建字符缓冲输入流对象
字符缓冲流特有功能
BufferedWriter:
void newLine()写一行行分隔符,行分隔符字符串由系统属性定义
BufferedReader:
String readLine()读一行文字。 结果包含行的内容的字符串,不包括任何行终止字符如果流的结尾已经到达,则为null
相关示例:
使用Java的输入、输出流将一个文本文件的内容按行读出,每读出一行就顺序添加行号,并写入到另一个文件中。
import java.io.*;
public class NumTest {
public static void main(String[] args) throws IOException {
//创建字符缓冲输入流
BufferedReader br = new BufferedReader(new FileReader("E:\\itcast\\java.txt"));
//创建字符缓冲输出流
BufferedWriter bw = new BufferedWriter(new FileWriter("E:\\itcast\\java1.txt"));
//更改数据并复制
String line;
int num = 0;
char[] chs=new char[1024];
while ((line=br.readLine())!=null){
//每行添加序号
line=++num+":"+line;
bw.write(line);
bw.newLine();
bw.flush();
}
//释放资源
bw.close();
br.close();
}
}
特殊流&Properties属性集
特殊流
标准输入流
System中有有两个静态的成员变量
public static final InputStream in:标准输入流。通常该流对应于键盘输入或由主机环境或用户指定的另一个输入源
public static final PrintStream out:标准输出流。通常该流对应于显示输出或由主机环境或用户指定的另一个输出目标
自己实现键盘录入数据
标准输出流
System类中有两个静态的成员变量
public static final InputStream in:标准输入流。通常该流对应于键盘输入或由主机环境或用户指定的另一个输入源
public static final PrintStream out:标准输出流。通常该流对应于显示输出或由主机环境或用户指定的另一个输出目标
输出语句的本质:是一个标准的输出流
- PrintStream ps = System.out;
- PrintStream类有的方法,System.out都可以使用
打印流
分类
字节打印流:PrintStream
- PrintStream(String fileName):使用指定的文件名创建新的打印流
- 使用继承父类的方法写数据,查看的时候会转码;使用自己的特有方法写数据,查看的数据原样输出
- 可以改变输出语句的目的地
public static void setOut(PrintStream out):重新分配“标准”输出流
字符打印流:PrintWriter
构造方法
PrintWriter(String fileName)使用指定的文件名创建一个新的PrintWriter,而不需要自动执行刷新
PrintWriter(Writer out, boolean autoFlush)创建一个新的PrintWriter out:字符输出流 autoFlush: 一个布尔值,如果为真,则println , printf ,或format方法将刷新输出缓冲区
特点
- 只负责输出数据,不负责读取数据
- 永远不会抛出IOException
- 有自己的特有方法
对象序列化流
对象序列化介绍
- 对象序列化:就是将对象保存到磁盘中,或者在网络中传输对象
- 这种机制就是使用一个字节序列表示一个对象,该字节序列包含:对象的类型、对象的数据和对象中存储的属性等信息
- 字节序列写到文件之后,相当于文件中持久保存了一个对象的信息
- 反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化
对象序列化流: ObjectOutputStream
将Java对象的原始数据类型和图形写入OutputStream。 可以使用ObjectInputStream读取(重构)对象。 可以通过使用流的文件来实现对象的持久存储。 如果流是网络套接字流,则可以在另一个主机上或另一个进程中重构对象
构造方法
ObjectOutputStream(OutputStream out)创建一个写入指定的OutputStream的ObjectOutputStream
序列化对象的方法
void writeObject(Object obj)将指定的对象写入ObjectOutputStream
注意事项
- 一个对象要想被序列化,该对象所属的类必须必须实现Serializable 接口
- Serializable是一个标记接口,实现该接口,不需要重写任何方法
对象反序列化流
对象反序列化流: ObjectInputStream
ObjectInputStream反序列化先前使用ObjectOutputStream编写的原始数据和对象
构造方法
ObjectInputStream(InputStream in)创建从指定的InputStream读取的ObjectInputStream
反序列化对象的方法
Object readObject()从ObjectInputStream读取一个对象
serialVersionUID&transient
serialVersionUID
- 用对象序列化流序列化了一个对象后,假如我们修改了对象所属的类文件,读取数据会不会出问题呢?
- 会出问题,会抛出InvalidClassException异常
- 如果出问题了,如何解决呢?
- 重新序列化
- 给对象所属的类加一个serialVersionUID
- private static final long serialVersionUID = 42L;
transient
如果一个对象中的某个成员变量的值不想被序列化,又该如何实现呢?
- 给该成员变量加transient关键字修饰,该关键字标记的成员变量不参与序列化过程
Properties集合
Properties作为Map集合
Properties介绍
- 是一个Map体系的集合类
- Properties可以保存到流中或从流中加载
- 属性列表中的每个键及其对应的值都是一个字符串
Properties作为Map集合的特有方法
Object setProperty(String key, String value): 设置集合的键和值,都是String类型,底层调用 Hashtable方法 put
String getProperty(String key):使用此属性列表中指定的键搜索属性
Set<String> stringPropertyNames(): 从该属性列表中返回一个不可修改的键集,其中键及其对应的值是字符串
Properties和IO流相结合的方法
void load(InputStream inStream)从输入字节流读取属性列表(键和元素对)
void load(Reader reader)从输入字符流读取属性列表(键和元素对)
void store(OutputStream out, String comments)将此属性列表(键和元素对)写入此 Properties表中,以适合于使用 load(InputStream)方法的格式写入输出字节流
void store(Writer writer, String comments)将此属性列表(键和元素对)写入此 Properties表中,以适合使用 load(Reader)方法的格式写入输出字符流
相关示例:
编写一个小程序,记录程序运行的次数,满足5次后,给出提示,试用次数已到,请注册! 分析: 1.将配置文件封装成File对象,判断文件是否存在,不存在则自己创建 2.需要计数器; 3.计数器的值,生命周期要比应用程序的生命周期要长,需要对计数器的值进行持久化。count = 1,里面存储的应该是键值方式,Map集合,要和设备上的数据关联,需要IO技术。集合 + IO = Properties。
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.Properties;
public class Test {
public static void main(String[] args) throws IOException {
//将配置文件封装成File对象。因为要判断文件是否存在。
File file = new File(
"idea_test[关卡8--16]\src\work14\SpecialstreamAndPropertiespropertyset\lianxi4\count.txt");
if (!file.exists()) {//如果不存在就创建
file.createNewFile();
}
//计数
int count = 0;//记录每次存储的次数
//创建集合对象
Properties prop = new Properties();
//定义输入流对象
FileInputStream fis = new FileInputStream(file);
//将流中的数据加载到集合中。
prop.load(fis);
//获取键对应的次数。
String s = prop.getProperty("count");
if (s != null) {
count = Integer.parseInt(s);
if (count >= 5) {
System.out.println("试用次数已到,请注册!");
}
}
count++;//对取出的次数进行自增。
//将键count,和自增后值重新存储到集合中。
prop.setProperty("count", Integer.toString(count));
//将集合中的数据存储到配置文件中。
FileOutputStream fos = new FileOutputStream(file);
prop.store(fos, "");
//释放资源
fos.close();
fis.close();
}
}