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 delet() |
删除由此抽象路径名表示的文件或目录 |
删除目录时的注意事项:
绝对路径和相对路径的区别
E:\\jihua\\java.txt
myFile\\java.txt
方法名 | 说明 |
---|---|
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对象数组 |
递归概述:以编程的角度来看,递归指的是方法定义中调用方法本身的现象
递归解决问题的思路:
把一个复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解
递归策略只需少量的程序就可描述出解题过程所需要的多次重复计算
递归解决问题要找到两个内容:
IO流概述:
IO流分类:
一般来说,我们说IO流的分类是按照数据类型来分的
那么这两种流都在什么情况下使用呢?
2.2字节流写数据
字节流抽象基类
lnputStream
:这个抽象类是表示字节输入流的所有类的超类OutputStream
:这个抽象类是表示字节输出流的所有类的超类FileOutputStream:文件输出流用于将数据写入File
FileOutputStream(Stringname)
:创建文件输出流以指定的名称写入文件使用字节输出流写数据的步骤:
方法名 | 说明 |
---|---|
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,则字节将写入文件的末尾而不是开头
finally:在异常处理时提供finally块来执行所有清除操作。比如说IO流中的释放资源
特点:被finally控制的语句一定会执行,除非JVM退出
try{
可能出现异常的代码;
}catch(异常类名变量名){
异常的处理代码;
}finally {
执行所有清除操作;
}
需求:把文件fos.txt中的内容读取出来在控制台输出
FilelnputStream
:从文件系统中的文件获取输入字节
FileInputStream(String name)
:通过打开与实际文件的连接来创建一个FilelnputStream
,该文件由文件系统中的路径名name命名使用字节输入流读数据的步骤:
①创建字节输入流对象
②调用字节输入流对象的读数据方法
int read()
:从该输入流读取一个字节的数据,当读取到文件末尾时返回-1③释放资源
//test.txt
hello
world
java
import java.io.FileInputStream;
import java.io.IOException;
public class demo {
public static void main(String[] args) throws IOException {
//创建字节输入流对象
FileInputStream fis = new FileInputStream("test.txt");
int re;
while ((re=fis.read())!=-1){
System.out.print((char)re);
}
//释放资源
fis.close();
}
}
//输出
hello
world
java
需求:把文件fos.txt中的内容读取出来在控制台输出
使用字节输入流读数据的步骤:
①创建字节输入流对象
②调用字节输入流对象的读数据方法
int read(byte[] b)
:从该输入流读取最多b.length个字节的数据到一个字节数组③释放资源
案例:复制图片
需求:把“D:\桌面\macOS 10.14-2022-04-23-16-17-25.png”复制到模块目录下的“mac.png"
思路:
代码:
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
public class demo {
public static void main(String[] args) throws IOException {
FileInputStream fis = new FileInputStream("D:\\桌面\\macOS 10.14-2022-04-23-16-17-25.png");
FileOutputStream fog = new FileOutputStream("mac.png");
byte[] by = new byte[1024];
int len;
while ((len=fis.read(by))!=-1){
fog.write(by,0,len);
}
fis.close();
fog.close();
}
}
字节缓冲流:
BufferOutputStream
:该类实现缓冲输出流。通过设置这样的输出流,应用程序可以向底层输出流写入字节,而不必为写入的每个字节导致底层系统的调用BufferedInputStream
:创建BufferedInputStream
将创建构造方法:
BufferedOutputStream(OutputStream out)
BufferedIlnputStream(InputStream in)
为什么构造方法需要的是字节流,而不是具体的文件或者路径呢
由于字节流操作中文不是特别的方便,所以Java就提供字符流
用字节流复制文本文件时,文本文件也会有中文,但是没有问题,原因是最终底层操作会自动进行字节拼接成
中文,如何识别是中文的呢?
基础知识:
字符集:
是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等
计算机要准确的存储和识别各种字符集符号,就需要进行字符编码,一套字符集必然至少有一套字符编码。
常见字符集有ASCIl字符集、GBXXX字符集、Unicode字符集等
ASCII字符集:
ASCI(American Standard Code for Information Interchange美国信息交换标准代码):是基于拉丁
字母的一套电脑编码系统,用于显示现代英语,主要包括控制字符(回车键、退格、换行键等)和可显示字符(英文大小写字符、阿拉伯数字和西文符号)
基本的ASCII字符集,使用7位表示一个字符,共128字符。ASCII的扩展字符集使用8位表示一个字符,共256字符,方便支持欧洲常用字符。是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等。
GBXXX字符集:
Unicode字符集:
为表达任意语言的任意字符而设计,是业界的一种标准,也称为统一码、标准万国码。它最多使用4个字节的数字来表达每个字母、符号,或者文字。有三种编码方案,UTF-8、UTF-16和UTF32。最为常用的UTF-8编码
UTF-8编码:可以用来表示Unicode标准中任意字符,它是电子邮件、网页及其他存储或传送文字的应用中,优先采用的编码。互联网工程工作小组(IETF)要求所有互联网协议都必须支持UTF-8编码。它使用一至四个字节为每个字符编码
小结:采用何种规则编码,就要采用对应规则解码,否则就会出现乱码
编码:
bytelgetBytes()
:使用平台的默认字符集将该String编码为一系列字节,将结果存储到新的字节数组中byte getBytes(String charsetName)
:使用指定的字符集将该String编码为一系列字节,将结果存储到新的字节数组中解码:
String(byte[] bytes)
:通过使用平台的默认字符集解码指定的字节数组来构造新的StringString(byte[] bytes, String charsetName)
:通过指定的字符集解码指定的字节数组来构造新的String字符流抽象基类
Reader
:字符输入流的抽象类Writer
:字符输出流的抽象类字符流中和编码解码问题相关的两个类:·
lnputStreamReader
OutputStreamWriter
示例:
import java.io.*;
public class demo {
public static void main(String[] args) throws IOException {
OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("./test.txt"), "GBK");
osw.write("你好世界");
osw.close();
InputStreamReader isr = new InputStreamReader(new FileInputStream("./test.txt"), "GBK");
char[] ch =new char[1024];
int len;
while ((len = isr.read(ch))!=-1) {
System.out.print(new String(ch,0,len));
}
isr.close();
}
}
//输出
你好世界
进程已结束,退出代码0
方法名 | 说明 |
---|---|
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(Rader in)
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
public class demo {
public static void main(String[] args) throws IOException {
BufferedReader bfr = new BufferedReader(new FileReader("test.txt"));
char[] ch = new char[1024];
int len;
while ((len = bfr.read(ch)) != -1) {
System.out.println(new String(ch, 0, len));
}
bfr.close();
}
}
void newLine()
:写一行行分隔符,行分隔符字符串由系统属性定义public String readLine()
:读一行文字。结果包含行的内容的字符串,不包括任何行终止字符(换行符),如果流的结尾已经到达,则为null
读数据
写数据
void write(int by):一次写一个字节
void write(byte[] bys, int index, int len):一次写一个字节数组的一部分
小结:字节流可以复制任意文件数据,有4种方式一般采用字节缓冲流一次读写一个字节数组的方式
读数据
int read()
:一次读取一个字符
int read(char[] chs)
:一次读取一个字符数组
写数据
void write(int ch)
:一次写一个字符
void write(char[] chs,int index,int len)
:一次写一个字符数组的一部分
String readLine():一次读取一个字符串
void newLine():写一个换行符
void write(String line):一次写一个字符串
小结:字符流只能复制文本数据,有5种方式,一般采用字符缓冲流的特有功能
try {
可能出现异常的代码;
} catch(异常类名变量名){
异常的处理代码;
} finally{
执行所有清除操作;
}
//try...catch...finally
private static void method() {
FileReader fr = null;
FileWriter fw = null;
try {
fr = new FileReader("test.txt");
fw = new FileWriter("test.txt");
char[] ch = new char[1024];
int len;
while ((len = fr.read(ch)) != -1) {
fw.write(ch, 0, len);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fr != null) {
try {
fr.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (fw != null) {
try {
fw.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
try(定义流对象){
可能出现异常的代码;
}catch(异常类名 变量名){
异常的处理代码;
}
//自动释放资源
//JDK7改进方案
private static void method() {
try (FileReader fr = new FileReader("test.txt");
FileWriter fw = new FileWriter("test.txt")) {
char[] ch = new char[1024];
int len;
while ((len = fr.read(ch)) != -1) {
fw.write(ch, 0, len);
}
} catch (IOException e) {
e.printStackTrace();
}
}
定义输入流对象;
定义输出流对象;
try(输入流对象;输出流对象){
可能出现异常的代码;
} catch(异常类名变量名){
异常的处理代码;
}
//自动释放资源
//JDK9改进方案
private static void method() throws IOException {
FileReader fr = new FileReader("test.txt");
FileWriter fw = new FileWriter("test.txt");
try (fr;fw) {
char[] ch = new char[1024];
int len;
while ((len = fr.read(ch)) != -1) {
fw.write(ch, 0, len);
}
} catch (IOException e) {
e.printStackTrace();
}
}
System类中有两个静态的成员变量:
public static final lnputStream in
:标准输入流。通常该流对应于键盘输入或由主机环境或用户指定的另一个输入源public static final PrintStream out
:标准输出流。通常该流对应于显示输出或由主机环境或用户指定的另一个输出目标自己实现键盘录入数据:
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
写起来太麻烦,Java就提供了一个类实现键盘录入
Scanner sc = new Scanner(System.in);
System类中有两个静态的成员变量:
public static final lnputStream in
:标准输入流。通常该流对应于键盘输入或由主机环境或用户指定的另一个输入源publicstatic final PrintStream out
:标准输出流。通常该流对应于显示输出或由主机环境或用户指定的另一个输出目标输出语句的本质:是一个标准的输出流
打印流分类:
PrintStream
PrintWriter
打印流的特点:
字节打印流
PrintStream(String fileName)
:使用指定的文件名创建新的打印流public static void main(String[] args) throws IOException {
PrintStream ps = new PrintStream("test.txt");
ps.write(97);
ps.print(97);
ps.println(98);
ps.println(99);
}
//test.txt
a9798
99
PrintWriter
,而不需要自动执行刷新|PrintWriter
out
:字符输出流autoFlush
:一个布尔值,如果为真,则printIn , printf,或format方法将刷新输出缓冲区|public class demo {
public static void main(String[] args) throws IOException {
PrintWriter ps = new PrintWriter("test.txt");
ps.write(97);
ps.print(97);
ps.println(98);
ps.println("你好");
ps.flush();
}
}
//test.txt
a9798
99
自动flush,使用第二个构造方法
public class demo {
public static void main(String[] args) throws IOException {
PrintWriter ps = new PrintWriter(new FileWriter("test.txt"),true);
ps.println("hello");
ps.println("world");
}
}
//test.txt
hello
world
对象序列化:就是将对象保存到磁盘中,或者在网络中传输对象
这种机制就是使用一个字节序列表示一个对象,该字节序列包含:对象的类型、对象的数据和对象中存储的属性等信息
字节序列写到文件之后,相当于文件中持久保存了一个对象的信息
反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化
要实现序列化和反序列化就要使用对象序列化流和对象反序列化流:
ObjectOutputStream
ObjectlnputStream
对象序列化流:ObjectOutputStream
OutputStream
。可以使用ObjectInputStream
读取(重构)对象。可以通过使用流的文件来实现对象的持久存储。如果流是网络套接字流,则可以在另一个主机上或另一个进程中重构对象构造方法:
ObjectOutputStream(OutputStream out)
:创建一个写入指定的OutputStream
的ObjectOutputStream
序列化对象的方法:
void writeObject(Object obj)
:将指定的对象写入ObjectOutputStream
注意:
一个对象要想被序列化,该对象所属的类必须必须实现**Serializable
**接口
Serializable
是一个标记接口,实现该接口,不需要重写任何方法
对象反序列化流:ObjectlnputStream
ObjectInputStream
反序列化先前使用ObjectOutputStream
编写的原始数据和对象构造方法:
ObjectlnputStream(InputStream in)
:创建从指定的lnputStream
读取的ObjectInputStream
反序列化对象的方法:
Object readObject()
:从ObjectlnputStream
读取一个对象示例:
//Student类
package com.demo;
import java.io.Serializable;
public class Student implements Serializable {
String name;
int age;
public Student() {
}
public Student(String name, int age) {
this.name = name;
this.age = age;
}
}
package com.demo;
import java.io.*;
public class demo {
public static void main(String[] args) throws IOException, ClassNotFoundException {
//创建对象序列化流
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("test.txt"));
//创建学生对象
Student s1 = new Student("张三", 16);
//序列化学生对象
oos.writeObject(s1);
//关闭对象序列化流
oos.close();
//创建对象反序列化流
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("test.txt"));
//反序列化学生对象
Object o = ois.readObject();//抛出ClassNotFoundException
//向下转型为学生类
Student s2 = (Student) o;
//输出学生类
System.out.println(s2.name+","+s2.age);//输出:张三,16
//关闭对象反序列化流
ois.close();
}
}
用对象序列化流序列化了一个对象后,假如我们修改了对象所属的类文件,读取数据会不会出问题呢?
InvalidClassException
异常如果出问题了,如何解决呢?
serialVersionUID
**private static final long serialVersionUID = 42L;
如果一个对象中的某个成员变量的值不想被序列化,又该如何实现呢?
Properties概述:
Properties
可以保存到流中或从流中加载//Properties作为Map集合的使用
import java.util.Properties;
public class demo {
public static void main(String[] args) {
//创建Properties集合
// Properties pro = new Properties();//错误,Properties默认使用Object类
Properties ppt = new Properties();
//存储元素
ppt.put("s1", "张三");
ppt.put("s2", "李四");
ppt.put("s3", "王五");
ppt.put("s4", "赵六");
//通过键遍历集合
for (Object key : ppt.keySet()) {
System.out.println(key + "," + ppt.get(key));
}
}
}
Properties
作为集合的特有方法:
方法名 | 说明 |
---|---|
Object setProperty(String key, String value) |
设置集合的键和值,都是String 类型,底层调用Hashtable 方法put |
String getProperty(String key) |
使用此属性列表中指定的键搜索属性 |
Set |
从该属性列表中返回一个不可修改的键集,其中键及其对应的值是字符串 |
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) 方法的格式写入输出字符流 |
示例:
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.util.Properties;
public class demo {
public static void main(String[] args) throws IOException {
save();
load();
}
private static void save() throws IOException {
//创建Properties集合
Properties ppt = new Properties();
//存储元素
ppt.setProperty("s1", "张三");
ppt.setProperty("s2", "李四");
ppt.setProperty("s3", "王五");
ppt.setProperty("s4", "赵六");
//存储集合
FileWriter fw = new FileWriter("test.txt");
ppt.store(fw, null);
fw.close();
}
private static void load() throws IOException {
//创建Properties集合
Properties ppt = new Properties();
//加载集合
FileReader fr = new FileReader("test.txt");
ppt.load(fr);
fr.close();
//输出集合
for (Object key : ppt.keySet()) {
System.out.println(key + "," + ppt.get(key));
}
}
}