学习笔记之IO流
IO流
File
File类概述
File:它是文件和目录路径名的抽象表示
- 文件和目录是可以通过File封装成对象的
- 对于File而言,其封装性并不是一个真正存在的文件,仅仅是一个路径名而已。它可以是存在的,也可以是不存在的,将来要是用过具体的操作把这个路径的内容转换为具体存在的
构造方法
- File(String pathname) 通过将给定的路径名字符串转换为抽象路径名来创建新的File实例
- File(String parent,String child) 从路径名字字符串和子路径名字字符串创建新的File实例
- File(File parent,String child)从父抽象路径名和子路径名字字符串创建新的File实例
import java.io.File;
public class Demo {
public static void main(String[] args) {
//File(String pathname) 通过将给定的路径名字符串转换为抽象路径名来创建新的File实例
File f1 = new File("D:\\JAVA\\java.txt");
System.out.println(f1);
//File(String parent,String child) 从路径名字字符串和子路径名字字符串创建新的File实例
File f2 = new File("D:\\JAVA","java.txt");
System.out.println(f2);
//File(File parent,String child)从父抽象路径名和子路径名字字符串创建新的File实例
File f3 = new File("D:\\JAVA");
File f4 = new File(f3,"java.txt");
System.out.println(f4);
}
}
File类创建功能
-
public boolean createNewFile() 当具有该名称的文件不存在时,创建一个由该抽象路径命名的新空文件
- 文件名存在,就不创建,返回false
- 文件名不存在,就创建,返回ture
-
public boolean mkdir() 创建由此路径名命名的目录
- 目录名存在,就不创建,返回false
- 目录名不存在,就创建,返回ture
-
punlic boolean mkdirs() 创建由此抽象路径名命名的目录,包括任何必须但不存在的父目录
- 目录名存在,就不创建,返回false
- 目录名不存在,就创建,返回ture
import java.io.File;
import java.io.IOException;
public class Demo {
public static void main(String[] args) throws IOException {
//在D:\\JAVA目录下创建一个文件java.txt
File f1 = new File("D:\\JAVA\\java.txt");
System.out.println(f1.createNewFile());
//在E:\\JAVA目录下创建一个目录JavaEE
System.out.println("----------------");
File f2 = new File("D:\\JAVA\\JAVAEE");
System.out.println(f2.mkdir());
//在E:\\JAVA目录下创建一个多级目录javaWEB\\HTML
System.out.println("----------------");
File f3 = new File("D:\\JAVA\\javaWEB\\HTML");
System.out.println(f3.mkdirs());
}
}
File判断和获取功能
- public boolean isDiectory() 测试此抽象路径名表示的File是否为目录
- public boolean isFiel() 测试此抽象路径名表示的Fiel是否为文件
- public boolean exists() 测试此抽象路径名表示的File是否存在
- public String getAbsolutePath() 返回此抽象路径名转换为路径名字字符串
- public String getPath() 将此抽象路径名转换为路径名字符串
- public String getName() 返回由此抽象路径名表示的文件或目录的名称
- public String[] list() 返回此抽象路径名表示的目录中的文件和目录的名称字符串数组
- public File[] listFiles() 返回此抽象路径名表示的目录中的文件和目录的File对象数组
import java.io.File;
public class Demo {
public static void main(String[] args) {
// 1. public boolean isDiectory() 测试此抽象路径名表示的File是否为目录
// 1. public boolean isFiel() 测试此抽象路径名表示的Fiel是否为文件
// 3. public boolean exists() 测试此抽象路径名表示的File是否存在
File f1 = new File("D:\\JAVA\\java.txt");
System.out.println(f1.isDirectory());
System.out.println(f1.isFile());
System.out.println(f1.exists());
System.out.println("---------------------------");
// 4. public String getAbsolutePath() 返回此抽象路径名转换为路径名字字符串
// 5. public String getPath() 将此抽象路径名转换为路径名字符串
// 6. public String getName() 返回由此抽象路径名表示的文件或目录的名称
System.out.println(f1.getAbsolutePath());
System.out.println(f1.getPath());
System.out.println(f1.getName());
// 7. public String[] list() 返回此抽象路径名表示的目录中的文件和目录的名称字符串数组
// 8. public File[] listFiles() 返回此抽象路径名表示的目录中的文件和目录的File对象数组
File f2 = new File("D:\\JAVA");
String[] list = f2.list();
System.out.println(list);
for (String li : list) {
System.out.println(li);
}
System.out.println("---------------------");
File[] files = f2.listFiles();
for (File filelist : files) {
// System.out.println(filelist);
if (filelist.isFile()) {
System.out.println(filelist.getName());
}
}
}
}
File类删除功能
- public boolean delete() 删除由此抽象路径名表示的文件或目录
遍历目录
/*
给定一个路径(E:\\JAVA),请通过递归完成遍历该目录下的所有内容,并把所有文件的绝对路径输出到控制台
思路:
根据给定路径创建一个File对象
定义一个方法用于获取给定目录下的所有内容,参数为第一步创建的File对象
获取给定的File目录下所有的文件或者目录的File数组
遍历该File数组,得到每一个File对象
判断该File对象是否是目录
是:递归调用
不是:获取绝对路径输出在控制台
调用方法
*/
import java.io.File;
public class Demo {
public static void main(String[] args) {
File fiel = new File("D:\\JAVA");
getAllFilePath(fiel);
}
public static void getAllFilePath(File file){
File[] files = file.listFiles();
if (files != null){
for (File i : files){
if (i.isDirectory()){
getAllFilePath(i);
}else{
System.out.println(i.getAbsolutePath());
}
}
}
}
}
IO流概述和分类
IO流概述:
- IO:输入/输出(input/Output)
- 流 : 是一种抽象概念,是对数据传输的总称。也就是说数据在设备间的传输称为流,流的本质是数据传输
- IO流就是用来处理设备间数据传输问题的
- 常见的应用:文件复制;文件上传;文件下载
分类
- 按数据流向:
- 输入流:读数据
- 输出流:写数据
- 按照数据类型分
- 字节流
- 字节输入流;字节输出流
- 字符流
- 字符输入流;字符输出流
- 字节流
- 一般IO流按照数据类型来分
使用场景:
- 如果数据通过Windows自带的记事本,我们还可以读懂里面的内容,用字符流;否则用字节流。若不知道使用哪种类型,用字节流(万能流)。
字节流
字节流抽象基类
- InputStream:这个抽象类是表示字节输入流的所有类的超类
- OutputStream:这个抽象类表示字节输出流的所有类的超类
- 子类名特点:子类名称都是以其父类名作为子类名的后缀
FileOutputStream:文件输出流用于将数据写入File
- FileOutputStream(String name):创建文件输出流以指定的名称写入文件
使用字节输出流写数据的步骤
- 创建字节输出流对象(调用系统功能创建了文件,创建字节输出流对象,让字节输出流对象指向文件)
- 调用字节输出流对象的写数据方法
- 释放资源(关闭此文件输出流并释放与此相关联的任何系统资源)
/*
FileOutputStream(String name):创建文件输出流以指定的名称写入文件
做了三件事:
调用系统功能创建了文件
创建了字节输出流对象
让字节输出流对象指向创建好的文件
void write(int b):将指定的字节写入此文件输出流
最后都要释放资源
void colse():关闭此文件输出流并释放与此相关联的任何系统资源
*/
import java.io.FileOutputStream;
import java.io.IOException;
public class Demo {
public static void main(String[] args) throws IOException {
FileOutputStream fos = new FileOutputStream("fos.txt");
fos.write(97);
fos.close();
}
}
字节流写数据的三种方式
- void write(int b) 将指定的字节写入此文件输出流
- void write(byte[] b)将b.length字节从指定的字节数组写入此文件输出流一次写一个字节数组数据
- void write(byte[] b, int off, int len)将len字节从指定的字节数组开始,从偏移量off(索引位置)开始写入此文件输出流一次写一个字节数组的部分数据
import java.io.FileOutputStream;
import java.io.IOException;
public class Demo {
public static void main(String[] args) throws IOException {
FileOutputStream fos = new FileOutputStream("fos.txt");
//void write(int b) 将指定的字节写入此文件输出流
// fos.write(97);
// fos.write(98);
// fos.write(99);
// fos.write(100);
// fos.write(101);
//void write(byte[] b)将b.length字节从指定的字节数组写入此文件输出流一次写一个字节数组数据
byte[] bytes = "abcde".getBytes();
// fos.write(bytes);
//void write(byte[] b, int off, int len)将len字节从指定的字节数组开始,从偏移量off(索引位置)开始写入此文
fos.write(bytes,1,3);
}
}
fos.colse();
写入数据的两个问题
- 如何换行
- Windows: \r\n
- Linux: \n
- mac: \r
- 如何追加写入
- public FileOutputStream(String name,boolean append)
创建文件输出流以指定的名称写入文件。
如果第二个参数为true ,则字节将写入文件的末尾而不是开头。
- public FileOutputStream(String name,boolean append)
import java.io.FileOutputStream;
import java.io.IOException;
public class Demo {
public static void main(String[] args) throws IOException {
FileOutputStream fos = new FileOutputStream("IO流\\fos.txt",true);
//如何换行
for (int i = 0; i < 10; i++){
fos.write("hello".getBytes());
//如何换行
fos.write("\r\n".getBytes());
}
//释放资源
fos.close();
}
}
字节流写数据+异常处理
- finally:在异常处理时提供finally块来执行所有请出操作。比如说IO流中的释放资源
- 特点:被finally控制的语句一定会执行,除非JVM退出
- 格式:
try{
可能出现异常的代码
}catch(异常类名 变量名){
异常的处理代码;
}finally{
执行所有清除操作;
}
import java.io.FileOutputStream;
import java.io.IOException;
public class Demo {
public static void main(String[] args) {
FileOutputStream fos = null;
try{
fos = new FileOutputStream("IO流\\fos.txt");
fos.write("hello".getBytes());
}catch (IOException e){
e.printStackTrace();
}finally {
if (fos != null){
try{
fos.close();
}catch (IOException e){
e.printStackTrace();
}
}
}
}
}
字节流读数据
FileInputStream:文件系统中的获取输入字节
- int read(),从该输入流读取一个字节的数据
- int read(byte[] b) 从该输入流读取最多 b.length个字节的数据到一个字节数组。
- int read(byte[] b, int off, int len) 从该输入流读取最多 len个字节的数据到字节数组。
- 使用字节输入流读数据的步骤:
- 创建字节输入流对象
- 调用字节输入流对象的读数据方法
- 释放资源
字节缓冲流
- BufferedOutputStream:该类实现缓冲输出流。通过设置这样的输出流,应用程序可以向底层输出流写入字节,而不必为写入的每个字节导致底层系统的调用
- BufferedInputStream:创建BufferedInputStream将创建一个内部缓冲区数组。当从流中读取或者跳过字节时,内部缓冲区将根据需要从所包含的输入流中重新填充,一次很多字节。
构造方法
- 字节缓冲输出流:BufferedOutputStream(OutputStream out)
- 字节缓冲输入流:BufferedInputStream(InputStream in)
为什么构造方法需要的是字节流,而不是具体的文件或者路径呢?
- 字节缓冲流仅仅提供缓冲区,而真正的读写数据还得依靠基本的字节流对象进行操作。
package 字节流.字节缓冲流;
import java.io.*;
/*
* BufferedOutputStream
* * BufferedInputStream
* 构造方法:
* 字节缓冲输出流:BufferedOutputStream(OutputStream out)
* 字节缓冲输入流:BufferedInputStream(InputStream in)
*/
public class Demo {
public static void main(String[] args) {
//BufferedOutputStream
/* BufferedOutputStream bos = null;
try {
bos = new BufferedOutputStream(new FileOutputStream("bos.txt"));
bos.write("hello\r\n".getBytes());
bos.write("world\r\n".getBytes());
}catch (IOException e){
e.printStackTrace();
}finally {
if (bos != null){
try {
bos.close();
}catch (IOException e){
e.printStackTrace();
}
}
}*/
//BufferedInputStream
//读一
/* BufferedInputStream bis = null;
try {
bis = new BufferedInputStream(new FileInputStream("bos.txt"));
int len;
while((len = bis.read()) != -1){
System.out.print((char)len);
}
}catch (IOException e){
e.printStackTrace();
}finally {
if (bis != null){
try {
bis.close();
}catch (IOException e){
e.printStackTrace();
}
}
}*/
//读二
BufferedInputStream bis = null;
try {
bis = new BufferedInputStream(new FileInputStream("bos.txt"));
byte [] b = new byte[1024];
int len;
while ((len = bis.read(b)) != -1){
System.out.print(new String(b,0,len));
}
}catch (IOException e){
e.printStackTrace();
}finally {
if (bis != null) {
try {
bis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
复制视频效率对比:
import java.io.*;
public class Demo {
public static void main(String[] args) {
long start = System.currentTimeMillis();
//缓冲字节流一次读取一个字节
// method(); // 50 ms
//缓冲字节流一次读取一个字节数组
method2(); // 2 ms
//基本字节流一次读取一个字节数组
// method3(); // 20 ms
//基本字节流一次读取一个字节
// method4(); // 6000 ms
long end = System.currentTimeMillis();
System.out.println(end - start + "ms");
}
//缓冲字节流一次读取一个字节
public static void method() {
BufferedInputStream bis = null;
BufferedOutputStream bos = null;
try {
bis = new BufferedInputStream(new FileInputStream("D:\\JAVA\\NBA.qlv"));
bos = new BufferedOutputStream(new FileOutputStream("D:\\JAVA\\JAVAEE\\NBA.qlv"));
int len;
while ((len = bis.read()) != -1) {
bos.write(len);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (bis == null || bos == null) {
try {
bis.close();
bos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
//缓冲字节流一次读取一个字节数组
public static void method2() {
BufferedInputStream bis = null;
BufferedOutputStream bos = null;
try {
bis = new BufferedInputStream(new FileInputStream("D:\\JAVA\\NBA.qlv"));
bos = new BufferedOutputStream(new FileOutputStream("D:\\JAVA\\JAVAEE\\NBA.qlv"));
byte[] b = new byte[1024];
int len;
while ((len = bis.read(b)) != -1) {
bos.write(b,0,len);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (bis == null || bos == null) {
try {
bis.close();
bos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
//基本字节流一次读取一个字节数组
public static void method3() {
FileOutputStream fos = null;
FileInputStream fis = null;
try {
fis = new FileInputStream("D:\\JAVA\\NBA.qlv");
fos = new FileOutputStream("D:\\JAVA\\JAVAEE\\NBA.qlv");
byte[] b = new byte[1024];
int len;
while ((len = fis.read(b)) != -1) {
fos.write(b,0,len);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fos == null || fis == null) {
try {
fos.close();
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
//基本字节流一次读取一个字节
public static void method4() {
FileOutputStream fos = null;
FileInputStream fis = null;
try {
fis = new FileInputStream("D:\\JAVA\\NBA.qlv");
fos = new FileOutputStream("D:\\JAVA\\JAVAEE\\NBA.qlv");
int len;
while ((len = fis.read()) != -1) {
fos.write(len);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fos == null || fis == null) {
try {
fos.close();
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
字节缓冲流效率明显高于基本字节流
字符流
为什么会出现字符流
由于字节流操作中文不是特别方便,所以Java就提供了字符流
- 字符流 = 字节流 + 编码表
用字节流复制文本文件式,文本文件也会有中文,但是没问题,是因为最终底层操作会自动进行字节拼接成中文,如何识别是中文呢? - 汉字在存储的时候,无论选择哪种编码存储,第一个字节都是负数。
编码:
- byte[] getBytes():使用平台默认字符集将该 String 编码为一系列字节,将结果存储到新的字节数组中
- byte[] getBytes(String charsetName):使用指定的字符集将该 String编码为一些列字节,将结果存储到新的字节数组中
解码:
- String(byte[] bytes):通过使用平台的默认字符集解码指定的字节数组来构造新的 String
- String(byte[] bytes, String charsetName):通过指定的字符集解码指定的字节数组来构造新的String
字符流编码解码问题
- 字符流抽象基类
- Reader:字符输入流的抽象类
- Writer:字符输出流的抽象类
- 字符流中和编码解码问题相关的两个类
- InputStreamReader:是从字节流到字符流的桥梁:它读取字节,并使用指定的charset将其解码为字符。 它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集。
- OutputStreamWriter:是从字符流到字节流的桥梁:使用指定的charset将写入的字符编码为字节。 它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集。
import java.io.*;
public class Demo {
public static void main(String[] args) throws IOException {
//编码
// OutputStreamWriter(OutputStream out) 创建一个使用默认字符编码的OutputStreamWriter。
// OutputStreamWriter(OutputStream out, String charsetName) 创建一个使用命名字符集的OutputStreamWriter。
//默认编码
OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("osw.txt"));
osw.write("中国");
osw.close();
//自定义编码
OutputStreamWriter osw2 = new OutputStreamWriter(new FileOutputStream("osw2.txt"), "GBK");
osw2.write("中华人民共和国");
osw2.close();
//解码
// InputStreamReader(InputStream in) 创建一个使用默认字符集的InputStreamReader。
// InputStreamReader(InputStream in, String charsetName) 创建一个使用命名字符集的InputStreamReader。
//默认编码
InputStreamReader isr = new InputStreamReader(new FileInputStream("osw.txt"));
int ch;
while ((ch = isr.read()) != -1){
System.out.print((char) ch);
}
isr.close();
System.out.println();
//自定义编码
InputStreamReader isr2 = new InputStreamReader(new FileInputStream("osw2.txt"),"GBK");
int len;
char [] c = new char[1024];
while((len = isr2.read(c)) != -1){
System.out.println(new String(c,0,len));
}
isr2.close();
}
}
字符流写数据的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)写入字符串的一部分
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
public class Demo {
public static void main(String[] args) throws IOException {
OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("osw.txt"));
//void write(int c) 写一个字符
osw.write(97);
osw.write(98);
//void write(char[] cbuf)写入一个字符数组
// char [] c = {'a','b','c','d'};
// osw.write(c);
//void write(char[] cbuf, int off, int len)写入字符数组的一部分
// char [] c = {'a','b','c','d'};
// osw.write(c,1,2);
//void write(String str)写一个字符串
// osw.write("hello");
//void write(String str, int off, int len)写入字符串的一部分
// osw.write("hello",1,2);
//void flush()刷新流
osw.flush();
osw.close();//关闭流 释放资源 先刷新 后释放
}
}
字符流读数据的两种方式
- int read()一次读一个字符数据
- int read(char[] cbuf)一次读一个字符数组数据
```import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStreamReader;
public class Demo {
public static void main(String[] args) throws IOException {
//int read()一次读一个字符数据
InputStreamReader isr = new InputStreamReader(new FileInputStream("osw.txt"));
int ch;
while ((ch = isr.read()) != -1){
System.out.print((char)ch);
}
isr.close();
System.out.println("--------------");
//int read(char[] cbuf)一次读一个字符数组数据
InputStreamReader isr2 = new InputStreamReader(new FileInputStream("osw.txt"));
int ch2;
char [] c = new char[1024];
while((ch2 = isr2.read(c)) != -1){
System.out.print(new String(c,0,ch2));
}
isr2.close();
}
}
转换流的名字较长,而我们常见的操作都是按照本地默认编码实现的,为了简便书写,转换流提供了对应的子类
- FileReader:用于读取取字符的便捷类
- FileReader(String fileName)
- FileWriter:用于写入字符文件的便捷类
- FileWriter(String fileName)
字符缓冲流
- BufferedReader:从字符输入流读取文本,缓冲字符,以提供字符,数组和行的高效读取。 可以指定缓冲区大小,或者可以使用默认大小。 默认值足够大,可用于大多数用途。
- BufferedWriter:将文本写入字符输出流,缓冲字符,以提供单个字符,数组和字符串的高效写入。 可以指定缓冲区大小,或者可以接受默认大小。 默认值足够大,可用于大多数用途。
构造方法:
- BufferedReader(Reader in)
- BufferedWriter(Writer out)
import java.io.*;
public class Demo {
public static void main(String[] args) throws IOException {
// BufferedWriter bw = new BufferedWriter(new FileWriter("bw.txt"));
// bw.write("hello\r\n");
// bw.write("world");
// bw.close();
BufferedReader br = new BufferedReader(new FileReader("bw.txt"));
//一次读取一个字符数据
// int len;
// while((len = br.read()) != -1){
// System.out.print((char)len);
// }
//一次读取一个字符数组
int len;
char [] c = new char[1024];
while((len = br.read(c)) != -1){
System.out.print(new String(c,0,len));
}
br.close();
}
}
字符缓冲流特有功能
- BufferedWriter
- void newLine():写一行行分隔符,行分隔符字符串由系统属性定义
- BufferedReader:
- public String readLine():读一行文字。结果包含行的内容的字符串,不包括任何行终止符,如果流的结尾已经到达,则为null
import java.io.*;
public class Demo {
public static void main(String[] args) throws IOException {
/* BufferedWriter bw = new BufferedWriter(new FileWriter("bw.txt"));
for (int i = 0; i < 10; i++){
bw.write("hello"+i);
bw.newLine();
}
bw.close();*/
BufferedReader br = new BufferedReader(new FileReader("bw.txt"));
String line;
while((line = br.readLine()) != null){
System.out.println(line);
}
br.close();
}
}
JDK7之后的异常处理
格式:
try(定义流对象){
可能出现异常的代码;
}catch(异常类 名变量名){
异常的处理代码;
}
自动释放资源
JDK9之后的异常处理
定义输入流对象;
定义输出流对象;
try(输入流对象;输出流对象){
可能出现异常的代码;
}catch(异常类 名变量名){
异常的处理代码;
}
自动释放资源
例如:
package 复制文件异常加入异常处理;
import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
public class Demo {
public static void main(String[] args) {
}
//JDK9的改进
private static void method3() throws IOException{
FileReader fr = new FileReader("fr.txt");
FileWriter fw = new FileWriter("fw.txt");
try(fr;fw){
int len;
char [] chars = new char[1024];
while ((len = fr.read(chars)) != -1){
fw.write(chars,0,len);
}
}catch (IOException e){
e.printStackTrace();
}
}
//JDK7的改进
private static void method2(){
try(FileReader fr = new FileReader("fr.txt");
FileWriter fw = new FileWriter("fw.txt")){
int len;
char [] chars = new char[1024];
while ((len = fr.read(chars)) != -1){
fw.write(chars,0,len);
}
}catch (IOException e){
e.printStackTrace();
}
}
//try...catch...finally
private static void method01(){
FileReader fr = null;
FileWriter fw = null;
try{
fr = new FileReader("fr.txt");
fw = new FileWriter("fw.txt");
int len;
char [] chars = new char[1024];
while ((len = fr.read(chars)) != -1){
fw.write(chars,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();
}
}
}
}
对象序列化流
对象序列化:就是将对象保存到磁盘中,或网络传输对象
这种机制就是使用一个字节序列表示一个对象,该字节序列包含:对象的类型、对象的数据和对象的存储的属性等信息字节序列写到文件之后,相当于文件中保存了一个对象的信息,反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化
- 对象序列化流:ObjectOutputStream
- 对象反序列化流:ObjectInputStream
对象序列化流
ObjectOutputStream将Java对象的原始数据类型和图形写入OutputStream。 可以使用ObjectInputStream读取(重构)对象。 可以通过使用流的文件来实现对象的持久存储。 如果流是网络套接字流,则可以在另一个主机上或另一个进程中重构对象。
构造方法:
- ObjectOutputStream(OutputStream out):创建一个写入指定的OutputStream的ObjiectOutputStream
序列化对象的方法:
- void writeObject(Object obj):将指定的对象写入ObjectOutputStream
注意:对象所属的类需要实现Serializable接口.不实现此接口将不会使任何状态序列化或反序列化(仅是一个标识接口,无需要重新的方法)
对象反序列化流
ObjectInputStream反序列化先前使用ObjectOutputStream编写的原始数据和对象。
构造方法:
- ObjectInputStream(InputStream in) 创建从指定的InputStream读取的ObjectInputStream。
反序列化对象的方法:
- Object readObject():从ObjectInputStream读取一个对象
- 用 transient 关键字修饰的成员变量不会被序列化
- 用对象序列流序列化了一个对象后,假如我们修改了对象所属的类文件,读取数据会抛出:InvalidClassException异常.我们可以给对象所属的类加一个serialVersionUID来解决这个问题。
private static final long serialVersionUID = 42L;
import java.io.Serializable;
public class Student implements Serializable {
private static final long serialVersionUID = 42L;
private String name;
private int age;
public Student() {
}
public Student(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectOutputStream;
public class Demo {
public static void main(String[] args) {
try(ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("oos.txt"))
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("oos.txt"))){
Student student = new Student("张三",20);
oos.writeObject(student);
Object obj = ois.readObject();
Student s = (Student)obj;
System.out.println(s.getName() + "," + s.getAge());
}catch (IOException | ClassNotFoundException e){
e.printStackTrace();
}
}
}
Properties
- 是一个Map体系的集合类
- Properties可以保存到流中或从流中加载
mport java.util.Properties;
import java.util.Set;
public class Demo {
public static void main(String[] args) {
Properties p = new Properties();
p.put("张三",20);
p.put("李四",18);
p.put("王五",19);
Set<Object> keys = p.keySet();
for (Object key : keys){
System.out.println(p.get(key));
}
}
}
特有方法:
- Objiec setProperty(String key,String value) 设值集合的键和值,都是String类型,底层调用Hashtable方法put
- String getProperty(String key) 使用此属性列表中指定的键搜索属性
- Set
stringPropertyNames() 从该属性列表返回一个不可修改的键集,其中键及其对应的值是字符串
import java.util.Properties;
import java.util.Set;
public class Demo2 {
public static void main(String[] args) {
//Objiec setProperty(String key,String value)
Properties p = new Properties();
p.setProperty("张三", "19");
p.setProperty("李四", "20");
p.setProperty("王五", "18");
System.out.println(p);
System.out.println("--------------");
//String getProperty(String key)
String zs = p.getProperty("张三");
System.out.println(zs);
System.out.println("--------------");
//Set<String> stringPropertyNames()
Set<String> keys = p.stringPropertyNames();
for (String key : keys) {
System.out.println(key);
}
System.out.println("------------------");
Set<String> names = p.stringPropertyNames();
for (String key : names) {
String values = p.getProperty(key);
System.out.println(key + "," + values);
}
}
}
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{
// mystore();
myload();
}
private static void mystore() throws IOException {
Properties p = new Properties();
p.setProperty("张三","20");
p.setProperty("李四","19");
p.setProperty("王五","18");
//void store(OutputStream out,String comments) comments描述
FileWriter fw = new FileWriter("IO流\\fw.txt");
p.store(fw,null);
fw.close();
}
private static void myload() throws IOException{
Properties p = new Properties();
FileReader fr = new FileReader("IO流\\fw.txt");
p.load(fr);
System.out.println(p);
}
}