什么是流?
流是内存和硬盘之间传输数据的通道。
水借助管道传输,数据借助流传输。
流的分类
按方向划分:
输入还是输出的划分参照物是内存
- 输入流:将硬盘的数据读取到内存中。
- 输出流:将内存中的数据写入到硬盘中。
按单位划分:
- 字节流:以字节为单位,可以读写所有数据。
- 字符流:以字符为单位,只能读写文本数据。
io流初识
Java中的io流都已经写好了,我们不需要关心底层源码,我们最主要的还是掌握在java中已经提供了哪些流,每个流的特点是什么?每个流对象上的常用方法有哪些?
java中所有的流都是在:java.io.*;下
我们主要还是研究:
- 怎么创建new流对象
- 调用流对象的哪个方法是读,哪个方法是写
java io流这块有四大家族:
四大家族的首领:
- java.io. Inputstream 字节输入流
- java.io. Outputstream 字节输出流
- java.io. Reader 字符输入流
- java. io. writer 字符输出流
四大家族的首领都是抽象类。
所有的流都实现了:
java.io. Closeable接口,都是可关闭的,都有close()方法。
流是一个管道,是内存和硬盘之间的通道,使用完之后一定要关闭流,不然会占用很多资源。
所有的输出流都实现了:
java.io.Flushable接口,都是可刷新的,都有flush()方法。
养成一个好习惯,输出流在最终输出之后,一定要记得flush()刷新一下。这个刷新表示将管道中剩余未输出的数据强制输出。
注意:
- 如果没有flush()可能会导致丢失数据。
- 在java中只要”类名“以Stream结尾的都是字节流,以Reader/Write结尾的都是字符流。
【6】java.io包下需要掌握的流有16个:
文件专属:
java. io. FileInputstream*
java. io Fileoutputstream*
java.io FileReader
java. io FileWriter
转换流:(将字节流转换成字符流)
java. io. InputstreamReader
java. io. Outputstreamwriter
缓冲流专属:
java. io BufferedReader
java. io Bufferedwriter
java. io BufferedInputstream
java. io. Bufferedoutputstream
数据流专属:
java. io DataInputstream
java. io DataOutputstream
标准输出流:
Java. io. Printwriter
java. io. Printstream
对象专属流:
java. io ObjectInputstream
java. io. ObjectOutputStream
【7】java.io.File类
File类的常用方法
【8】java io这块还剩下的内容:
- ObjectInputStream ObjectOutputStream的使用
- io流+Properties集合的联合使用
io流:文件的读和写
Properties:是一个Map集合,key和value都是String类型
字节流
字节流是抽象类:
文件字节流
字节输入流
java.io.FileInputStream:
FileInputStream对象是把数据从硬盘文件读取到内存。
1、文件字节输入流,任何类型的文件都可以用这个流来读取!
2、以字节的方式,完成读的操作。(硬盘-->内存)
package com.io.demo01;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
/*
java.io.FileInputStream:
1、文件字节输入流,任何类型的文件都可以用这个流来读取
2、字节的方式,完成读的操作。(硬盘-->内存)
*/
public class Demo01 {
public static void main(String[] args) {
FileInputStream fis = null;
try {
//创建一个FileInputStream对象
fis = new FileInputStream("C:\\Users\\Siming\\IdeaProjects\\JavaSE\\基础语法\\src\\com\\io\\demo01\\test2");
//从文件读取数据到内存
int readData = fis.read(); //读取一个字节,返回字节的ASCII码
System.out.println(readData);
int readData1 = fis.read();
System.out.println(readData1);
int readData2 = fis.read();
System.out.println(readData2);
int readData3 = fis.read();
System.out.println(readData3);
int readData4 = fis.read();
System.out.println(readData4);
int readData5 = fis.read();
System.out.println(readData5);
//已经读到文件末尾了,再读也读取不到任何数据,返回-1
int readData6 = fis.read();
System.out.println(readData6);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fis != null) { //避免空指针异常
try {
fis.close(); //关闭管道
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
上面这个程序有缺点:
一次读取一个字节byte,这样内存和硬盘的交互太频繁,基本上时间和资源都浪费在交互上面。
能不能一次读取多个字节呢?答案是可以。
一次读取多个字节:
package com.io.demo01;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
/*
这个程序的缺点:
一次读取一个字节byte,这样内存和硬盘的交互太频繁,基本上时间和资源都浪费在交互上面。
能不能一次读取多个字节呢?答案是可以。
*/
public class Demo02 {
public static void main(String[] args) {
FileInputStream fis = null;
try {
//创建FileInputStream对象
fis = new FileInputStream("C:\\Users\\Siming\\IdeaProjects\\JavaSE\\基础语法\\src\\com\\io\\demo01\\test2");
//从文件读取数据到内存,利用while循环一次读取多个字节数据
// while (true) {
// int readData = fis.read();
// if (readData == -1) {
// break;
// }
// System.out.println(readData);
// }
//改造while循环
int readDate = 0;
while ((readDate = fis.read()) != -1) {
System.out.println(readDate);
}
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fis != null) {
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
读到的数据存到字节数组:
package com.io.demo01;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
/*
int read(byte[] b)方法
一次最多读取b.length 个字节
这种读取方法可以减少硬盘和内存的交互,提高程序执行效率。
读到的数据存到byte[]数组
*/
public class Demo03 {
public static void main(String[] args) {
FileInputStream fis = null;
try {
//创建FileInputStream对象
//绝对路径
// fis = new FileInputStream("C:\Users\Siming\IdeaProjects\JavaSE\基础语法\src\com\io\demo01\test2");
//相对路径,IDEA默认的当前路径是工程Project的根路径
fis = new FileInputStream("基础语法\\src\\com\\io\\demo01\\test2");
//从文件读取数据到内存
byte[] bytes = new byte[4]; //创建一个长度为4的byte数组,一次最多可以读取4个字节
int readCount = fis.read(bytes); //返回的是读到的字节数量,不是字节的本身ASCII码
System.out.println(readCount); //第一次读到4个字节
//byte数组全部转换为String字符串
// System.out.println(new String(bytes));
//不应该全部转换,应该是读了多少个,转换多少个
System.out.println(new String(bytes, 0, readCount));
int readCount2 = fis.read(bytes); //第二次读到2个字节,新读到的字节把数组中的老字节覆盖
System.out.println(readCount2);
// System.out.println(new String(bytes));
System.out.println(new String(bytes, 0, readCount2));
int readCount3 = fis.read(bytes); //第三次一个字节也没读到,返回-1
System.out.println(readCount3);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fis != null) {
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
FileInputStream其他常用方法:available(),skip()
package com.io.demo01;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
public class Demo05 {
public static void main(String[] args) {
/*
FileInputStream其他常用方法:
available(),返回流中剩下没有读到的字节数量
skip(),跳过几个字节不读
*/
FileInputStream fis = null;
try {
//创建FileInputStream对象
fis = new FileInputStream("基础语法\\src\\com\\io\\demo01\\test2");
System.out.println("总字节数量是:"+fis.available());
// //读一个字节
// int readByte = fis.read();
// //剩下还可以读的数量是:5
// System.out.println(fis.available());
// //这个方法有什么用?
// byte[] bytes = new byte[fis.available()]; //这种方法不适合太大的文件,因为byte[]数组不能太大。
// //不需要循环了,直接一次读完
// int readCount = fis.read(bytes);
// System.out.println(new String(bytes));
//跳过几个字节不读
fis.skip(3);
System.out.println(fis.read()); //从第四个字节d开始读
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fis != null) {
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
字节输出流
FileOutputStream对象是把数据从内存写出到硬盘文件。
只能写出byte类型数据,一般使用字节数组存储要写出的数据,其他类型(String)想写出到文件的话,要转换为byte类型,再写出。
文件字节输入流,任何类型的文件都可以用这个流来读取!
以字节的方式,完成写的操作。(内存-->硬盘)
package com.io.demo01;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
public class Demo06 {
public static void main(String[] args) {
FileOutputStream fos = null;
try {
//创建FileOutputStream对象
//myFile文件不存在时会自动新建!
//这种方法谨慎小心使用,因为这种方法会先把原文件清空,然后再写入
// fos = new FileOutputStream("基础语法\\src\\com\\io\\demo01\\myFile");
//以追加的方式在文件末尾写入,不会清空原文件内容。
fos = new FileOutputStream("基础语法\\src\\com\\io\\demo01\\myFile", true);
//从内存写出到硬盘(文件)
byte[] bytes = {97, 98, 99, 100};
//将byte数组全部写出
fos.write(bytes); //abcd
//将byte数组一部分写出
fos.write(bytes, 0, 2); //ab,从下标0开始写出,共写出2个字节
//字符串
String str = "华为,不仅仅是世界500强。";
//将String转换为byte数组
byte[] bytes2 = str.getBytes();
//写操作
fos.write(bytes2);
//写完以后,一定要刷新管道
fos.flush();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fos != null) {
try {
fos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
使用字节流读写文件
使用FileInputStream + FileOutputStream 完成文件的拷贝。
拷贝的过程应该是一边读,一边写
使用以上的字节流拷贝文件的时候,任何文件类型都可以,什么都可以拷。
package com.io.demo01;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
/*
使用FileInputStream + FileOutputStream 完成文件的拷贝
拷贝的过程应该是一边读,一边写
使用以上的字节流拷贝文件的时候,任何文件类型都可以,什么都可以拷。
*/
public class Demo07 {
public static void main(String[] args) {
FileInputStream fis = null;
FileOutputStream fos = null;
try {
//创建字节输入流对象
fis = new FileInputStream("C:\\Users\\Siming\\Videos\\The Voice Kids.2019.精选.mp4");
//创建字节输出流对象
fos = new FileOutputStream("C:\\Users\\Siming\\Music\\The Voice Kids.2019.精选.mp4");
//核心代码:一边读,一边写
byte[] bytes = new byte[1024 * 1024]; //1MB,一次最多拷贝1Mb,1024是1kB
int readCount = 0;
while ((readCount = fis.read(bytes)) != -1) {
fos.write(bytes, 0, readCount);
}
//写完以后,要刷新管道
fos.flush();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
//分开try,不要一起try
//一起try时,其中一个出现异常,可能会影响另一个流的关闭
if (fos != null) {
try {
fos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (fis != null) {
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
文件字符流
字符输入流
java.io.FileReader:
FileReader对象是把文件中的字符从硬盘文件读取到内存。
字符输入流特点:只能读取普通文本,读取文本内容时,比较方便,快捷。
package com.io.demo01;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
/*
FileReader:
文件字符输入流,只能读取普通文本
读取文本内容时,比较方便,快捷
*/
public class Demo08 {
public static void main(String[] args) {
FileReader reader = null;
try {
//创建文件字符输入流
reader = new FileReader("基础语法\\src\\com\\io\\demo01\\test");
char[] chars = new char[4]; //创建一个长度为4的char数组,一次最多可以读取4个字符
//读到的字符存到char数组中
reader.read(chars);
//使用增强for循环遍历数组中的字符
for (char c : chars) {
System.out.print(c);
}
//连续读取字符
// char[] chars2 = new char[4]; //一次最多读取4个字符
int readCount = 0;
while ((readCount = reader.read(chars)) != -1) {
System.out.print(new String(chars, 0, readCount));
}
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (reader != null) {
try {
reader.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
字符输出流
java.io.FileWriter:
FileWriter对象是把字符数据从内存写出到硬盘文件。
FileWriter的特点:只能输出普通文本。
package com.io.demo01;
import java.io.FileWriter;
import java.io.IOException;
/*
FileWriter:
文件字符输出流。写操作
只能输出普通文本。
*/
public class Demo09 {
public static void main(String[] args) {
FileWriter fw = null;
try {
//创建文件字符输出流对象
fw = new FileWriter("基础语法\\src\\com\\io\\demo01\\test", true);
//写操作
char[] chars = {'早', '安', '打', '工', '人'};
fw.write(chars);
fw.write(chars, 2, 3);
fw.write("\n");
fw.write("华为,不仅仅是世界500强。");
fw.write("我是一名Java攻城狮");
//写完以后,要刷新管道
fw.flush();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fw != null) {
try {
fw.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
使用字符流读写文本文件
使用FileReader + FileWriter进行拷贝的话,只能拷贝普通文本文件
拷贝的过程应该是一边读,一边写
什么是文本文件?
能用记事本打开编辑的都是普通文本文件
package com.io.demo01;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
/*
使用FileReader + FileWriter进行拷贝的话,只能拷贝普通文本文件
能用记事本打开编辑的都是普通文本文件
*/
public class Demo10 {
public static void main(String[] args) {
FileReader fr = null;
FileWriter fw = null;
try {
//创建文本字符输入流,读
fr = new FileReader("基础语法\\src\\com\\io\\demo01\\Demo10.java");
//创建文本字符输出流,写
fw = new FileWriter("Demo10.java");
//核心代码:一边读,一边写
char[] chars = new char[1024 * 512]; //1MB
int readCount = 0;
while ((readCount = fr.read(chars)) != -1) {
fw.write(chars, 0, readCount);
}
//写完以后,要刷新管道
fw.flush();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fw != null) {
try {
fw.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (fr != null) {
try {
fr.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
缓冲字符流
缓冲字符输入流
java.io.BufferedReader:
BufferedReader对象是把文件中的字符从硬盘文件读取到内存。
BufferedReader的特点:
- 是个带有缓冲区的字符输入流
- 使用这个流的时候,不需要自定义char数组。自带缓冲区!
- readLine()方法读取一个文本行,但不带换行符!
什么是节点流?什么是包装流?
- 当一个流的构造方法中需要一个流的时候,这个被传进来的流叫做节点流。
- 外部负责包装的这个流叫做包装流。
- 对于包装流来说,只需要关闭最外层流就可以了,里面的节点流会自动关闭。
package com.io.demo01;
import java.io.BufferedReader;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
/*
BufferedReader:
带有缓冲区的字符输入流
使用这个流的时候,不需要自定义char数组或byte数组。自带缓冲。
*/
public class Demo11 {
public static void main(String[] args) throws IOException {
FileReader fr = null;
try {
//创建文件字符输出流对象
fr = new FileReader("基础语法\\src\\com\\io\\demo01\\Demo11.java");
} catch (FileNotFoundException e) {
e.printStackTrace();
}
//当一个流的构造方法中需要一个流的时候,这个被传进来的流叫做节点流。
//外部负责包装的这个流叫做包装流/处理流。
//在当前这个程序,FileReader是节点流,BufferedReader是包装流
////创建缓冲字符输出流对象
BufferedReader br = new BufferedReader(fr);
//读取字符,一次读一行字符串
// try {
// String firstLine = br.readLine();
// System.out.println(firstLine);
//
// String secondLine = br.readLine();
// System.out.println(secondLine);
//
// String thirdLine = br.readLine();
// System.out.println(thirdLine);
// } catch (IOException e) {
// e.printStackTrace();
// }
//br.readLine()方法读取一个文本行,但不带换行符
String s = null;
while ((s = br.readLine()) != null) {
System.out.println(s);
}
//关闭流
//对于包装流来说,只需要关闭最外层流就可以了,里面的节点流会自动关闭
br.close();
}
}
缓冲字符输出流
java.io.BufferedWriter:
BufferedWriter对象是将字节流转换为字符流。
BufferedWriter的特点:
- 是个带有缓冲区的字符输出流
- 使用这个流的时候,不需要自定义char数组。自带缓冲区!
package com.io.demo01;
import java.io.BufferedWriter;
import java.io.FileOutputStream;
import java.io.FileWriter;
import java.io.OutputStreamWriter;
/*
BufferedWrite:
带有缓冲区的字符输出流
使用这个流的时候,不需要自定义char数组。自带缓冲。
*/
public class Demo13 {
public static void main(String[] args) throws Exception {
//创建字符输出流对象
FileWriter fw = new FileWriter("基础语法\\src\\com\\io\\demo01\\test");
//创建BufferedWriter对象
BufferedWriter bw = new BufferedWriter(fw);
//使用转换流的BufferedWriter
// BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(new FileOutputStream("基础语法\\src\\com\\io\\demo01\\test")));
//写操作
char[] chars = {'a', 'b', 'c', 'd'};
String str = "华为,不仅仅是世界五百强!";
bw.write(chars);
bw.write(chars, 2,2);
bw.write("\n");
bw.write("HUAWEI");
bw.write(str);
bw.write(str, 0, 2);
//写完以后,要刷新管道
bw.flush();
//关闭最外层流
bw.close();
}
}
转换流
java.io.InputStreamReader:
InputStreamReader对象是将字节流转换为字符流。
package com.io.demo01;
import java.io.BufferedReader;
import java.io.FileInputStream;
import java.io.InputStreamReader;
/*
转换流:InputStreamReader
*/
public class Demo12 {
public static void main(String[] args) throws Exception {
//创建字节流输入对象
FileInputStream fis = new FileInputStream("基础语法\\src\\com\\io\\demo01\\Demo11.java");
//创建转换流对象
//通过转换流进行转换(InputStreamReader将字节流转成字符流)
//fis是节点流,ir是包装流
InputStreamReader ir = new InputStreamReader(fis);
//这个构造方法只能传一个字符流,不能传字节流
//ir是节点流,br是包装流
BufferedReader br = new BufferedReader(ir);
//合并
// BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream("基础语法\\src\\com\\io\\demo01\\Demo11.java")));
//读操作
String line = null;
while ((line = br.readLine()) != null) {
System.out.println(line);
}
//关闭最外层流
br.close();
}
}
数据流
数据字节输出流
java.io.DataOutputStream:
DataOutputStream是把数据和数据类型加密输出到文件中,一般方式无法打开这个文件。
package com.io.demo01;
import java.io.DataOutputStream;
import java.io.FileOutputStream;
/*
java.io.DataOutputStream:数据专属的流
这个流可以将数据连同数据的类型一并写入文件。
注意:这个文件不是普通文本文档。(用记事本打不开)
*/
public class Demo14 {
public static void main(String[] args) throws Exception {
//创建数据专属的字节输出流
DataOutputStream dos = new DataOutputStream(new FileOutputStream("基础语法\\src\\com\\io\\demo01\\test1"));
//准备数据
byte b = 100;
short s = 200;
int i = 300;
long l = 400L;
float f = 3.14f;
double d = 23.4;
boolean bl = false;
char c = 'a';
//写数据
dos.writeByte(b);
dos.writeShort(s);
dos.writeInt(i);
dos.writeLong(l);
dos.writeFloat(f);
dos.writeDouble(d);
dos.writeBoolean(bl);
dos.writeChar(c);
//写完以后,要刷新管道
dos.flush();
//关闭最外层流
dos.close();
}
}
数据字节输入流
java.io.DataInputStream:
DataInputStream对象把加密文件的内容解密出来,拿出存在里面数据和数据类型。
DataInputStream的特点:
- DataOutputSteam写的文件,只能使用DataInputSteam去读,并且读的时候你需要提前知道写入的顺序。读的顺序需要和写的顺序一致,才可以正常取出数据。
package com.io.demo01;
import java.io.DataInputStream;
import java.io.FileInputStream;
/*
DataInputSteam:数据字节输入流
DataOutputSteam写的文件,只能使用DataInputSteam去读,并且读的时候你需要提前知道写入的顺序。
读的顺序需要和写的顺序一致,才可以正常取出数据。
*/
public class Demo15 {
public static void main(String[] args) throws Exception {
//创建数据专属的字节输入流
DataInputStream dis = new DataInputStream(new FileInputStream("基础语法\\src\\com\\io\\demo01\\test1"));
//读数据
byte b = dis.readByte();
short s = dis.readShort();
int i = dis.readInt();
long l = dis.readLong();
float f = dis.readFloat();
double d = dis.readDouble();
boolean bl = dis.readBoolean();
char c = dis.readChar();
System.out.println(b);
System.out.println(s);
System.out.println(i + 1000);
System.out.println(l);
System.out.println(f);
System.out.println(d);
System.out.println(bl);
System.out.println(c);
//关闭最外层流
dis.close();
}
}
标准输出流
标准字节输出流
import java.io.PrintStream:
PrintStream对象是标准的字节输出流。
PrintStream的特点:
- 默认输出到控制台
- 可以把输出方向指向文件,输出内容保存到文件中。
package com.io.demo01;
import java.io.FileOutputStream;
import java.io.PrintStream;
/*
java.io.PrintSteam:标准的字节输出流,默认输出到控制台
*/
public class Demo16 {
public static void main(String[] args) throws Exception {
//联合起来写
System.out.println("Hello, World!");
//分开写
PrintStream ps = System.out; //标准输出流指向控制台
ps.println("Hello Java");
ps.println("Hello,Python");
ps.println("Hello,Tencent");
//标准输出流不再指向控制台,指向“log”文件
PrintStream printStream = new PrintStream(new FileOutputStream("基础语法\\src\\com\\io\\demo01\\log"));
//修改输出方向,将输出方向修改到“log”文件
System.setOut(printStream);
//再输出
System.out.println("word很大,你忍一下!");
System.out.println("蚌埠住了");
System.out.println("在一起,就可以!");
//标准输出流不需要手动close()关闭
}
}
日志输出
使用了标准字节输出流,把输出的日志信息保存在日志文件中。
程序入口:
package com.io.demo01;
public class LogTest {
public static void main(String[] args) {
//测试工具类是否好用
Logger.log("调用了System类的gc()方法,建议启动垃极回收");
Logger.log("调用了User Service的doSome()方法");
Logger.log("用户尝试进行登录,验证失败");
}
}
日志类:
package com.io.demo01;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.PrintStream;
import java.text.SimpleDateFormat;
import java.util.Date;
/*
日志工具Demo17
*/
public class Logger {
//记录日志的方法
public static void log(String msg) {
try {
//输出指向日志文件
PrintStream out = new PrintStream(new FileOutputStream("基础语法\\src\\com\\io\\demo01\\logFile.txt", true));
//改变输出方向
System.setOut(out);
//日期当前时间
Date nowTime = new Date();
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss SSS");
String strTime = sdf.format(nowTime);
System.out.println(strTime + ": " + msg);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
}
File类
java.io.File:
FiLe类不能完成文件的读和写。
File对象代表什么?
- 文件和目录路径名的抽象表示形式。
- C:\Users\Siming\Downloads\Compressed 是一个对象
- C:\Users\Siming\Downloads\Compressed\JDK_API_1_6_zh_CN.CHM 也是一个对象
- 一个File对象有可能对应的是目录,也可能是文件
- File只是一个路径名的抽象表示形式
package com.io.demo02;
import java.io.File;
/*
File
1.File类和四大家族没有关系,所以FiLe类不能完成文件的读和写。
2.File对象代表什么?
文件和目录路径名的抽象表示形式。
C:\Users\Siming\Downloads\Compressed 是一个对象
C:\Users\Siming\Downloads\Compressed\JDK_API_1_6_zh_CN.CHM 也是一个对象
一个File对象有可能对应的是目录,也可能是文件
File只是一个路径名的抽象表示形式。
需要掌握File类中常用的方法
*/
public class Demo01 {
public static void main(String[] args) throws Exception {
//创建一个File对象
File f1 = new File("C:\\Users\\Siming\\IdeaProjects\\JavaSE\\基础语法\\src\\com\\io\\demo01\\file");
//判断是否存在!
System.out.println(f1.exists());
//如果对象不存在,则以文件的形式创建出来
if (!f1.exists()) {
//以文件形式新建
f1.createNewFile();
}
//如果对象不存在,则以目录的形式创建出来
if (!f1.exists()) {
f1.mkdir();
}
//可以创建多重目录
File f2 = new File("C:\\Users\\Siming\\IdeaProjects\\JavaSE\\基础语法\\src\\com\\io\\demo01\\file\\a\\b\\c");
if (!f2.exists()) {
f2.mkdirs();
}
File f3 = new File("C:\\Users\\Siming\\IdeaProjects\\JavaSE\\基础语法\\src\\com\\io\\demo01\\log");
//获取文件的父路径
String parentPath = f3.getParent();
System.out.println(parentPath);
//获取文件的父文件对象
File parentFile = f3.getParentFile();
System.out.println("获取绝对路径:" + parentFile.getAbsolutePath());
File f4 = new File("C:\\Users\\Siming\\IdeaProjects\\JavaSE\\基础语法\\src\\com\\io\\demo01\\test");
System.out.println("绝对路径:" + f4.getAbsolutePath());
}
}
File类的常用方法:
package com.io.demo02;
import java.io.File;
import java.text.SimpleDateFormat;
import java.util.Date;
/*
FIle类的常用方法
*/
public class Demo02 {
public static void main(String[] args) {
//创建File对象
File f1 = new File("C:\\Users\\Siming\\IdeaProjects\\JavaSE\\基础语法\\src\\com\\io\\demo01\\log");
//获取文件名
System.out.println("文件名:" + f1.getName());
//判断是否是一个目录
System.out.println(f1.isDirectory()); //false
//判断是否是一个文件
System.out.println(f1.isFile()); //true
//获取文件最后一次修改时间
long haoMiao = f1.lastModified(); //这个毫秒是从1970年到现在的总毫秒数
//将总毫秒数转换成日期
Date time = new Date(haoMiao);
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss SSS");
String strTime = sdf.format(time);
System.out.println(strTime);
//获取文件大小
System.out.println(f1.length()); //55字节
}
}
File类的listFiles方法:
可以获取当前目录下的所有子文件。
package com.io.demo02;
import java.io.File;
/*
File中的listFiles方法
*/
public class Demo03 {
public static void main(String[] args) {
//File[] listFiles()
//获取当前目录下的所有子文件
File f1 = new File("C:\\Users\\Siming\\IdeaProjects\\JavaSE\\基础语法\\src\\com\\io\\demo01");
File[] files = f1.listFiles();
//增强for循环遍历
for (File file : files) {
System.out.println(file.getAbsolutePath());
System.out.println(file.getName());
}
}
}
拷贝目录
拷贝目录需要三要素:
- 拷贝源
- 拷贝目标
- 拷贝方法
拷贝方法是核心,大体思路是先建立文件夹,再往文件夹中拷贝文件。
package com.io.demo03;
import java.io.*;
/*
拷贝目录
*/
public class Demo01 {
public static void main(String[] args) {
//拷贝源
File srcFile = new File("C:\\Users\\Siming\\Downloads\\夏目友人帐-字幕");
//拷贝目标
File destFile = new File("C:\\Users\\Siming\\Videos\\");
//调用拷贝方法
copyDir(srcFile, destFile);
}
/**
* 拷贝目录
* @param srcFile 拷贝源
* @param destFile 拷贝目标
*/
private static void copyDir(File srcFile, File destFile) {
if (srcFile.isFile()) {
//srcFile如果是一个文件的话,递归结束
//是文件的时候需要拷贝
//...一边读,一边写
FileInputStream fis = null;
FileOutputStream fos = null;
try {
//读这个文件
//C:\Users\Siming\Downloads\夏目友人帐-字幕\夏目友人帐\1\Natsume Yuujinchou 01.ass
fis = new FileInputStream(srcFile);
//写到这个文件中
//C:\Users\Siming\Videos\夏目友人帐-字幕\夏目友人帐\1\Natsume Yuujinchou 01.ass
String path = (destFile.getAbsolutePath().endsWith("\\") ? destFile.getAbsolutePath() :
destFile.getAbsolutePath() + "\\") + srcFile.getAbsolutePath().substring(26);
fos = new FileOutputStream(path);
//一边读,一边写
byte[] bytes = new byte[1024 * 1024]; //一次复制1MB
int readCount = 0;
while ((readCount = fis.read(bytes)) != -1) {
fos.write(bytes, 0, readCount);
}
//写完以后,要刷新管道
fos.flush();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fis != null) {
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (fos != null) {
try {
fos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
return;
}
//获取源下面的子目录
File[] files = srcFile.listFiles();
for (File file : files) {
//获取所有文件的(包括目录和文件)绝对路径
// System.out.println(file.getAbsolutePath());
if (file.isDirectory()) {
//新建对应目录
// System.out.println(file.getAbsolutePath());
//C:\Users\Siming\Downloads\夏目友人帐-字幕\夏目友人帐 源目录
//C:\Users\Siming\Videos\夏目友人帐-字幕\夏目友人帐 目标目录
String srcDir = file.getAbsolutePath();
String destDir = (destFile.getAbsolutePath().endsWith("\\") ? destFile.getAbsolutePath() : destFile.getAbsolutePath() + "\\") + srcDir.substring(26);
File newFile = new File(destDir);
if (!newFile.exists()) {
newFile.mkdirs();
}
}
//递归调用
copyDir(file, destFile);
}
}
}
序列化
想要序列化对象,必须要实现Serializable接口。
Serializable接口的特点:
- Serializable接口只是一个标志接口,java虚拟机看到这个类实现了这个接口,就会给这个类特殊服务。
- Java虚拟机看到这个接口之后,会给这个类生成一个序列化版本号。
Java语言中是采用什么机制区分“类”的?
- 第一:首先通过类名进行比对,如果类名不一样,那肯定不是同一个类。
- 第二:如果类名一样,还能怎么区分类?靠序列化版本号进行区分。
最终结论:
凡是一个类实现了Serializable接口,建议给这个类提供一个固定不变的序列化版本号。这样,以后这个类的代码即使修改了,但是版本号不变,Java虚拟机还是会认为是同一个类。
序列化对象:
package com.io.demo04;
import java.io.FileOutputStream;
import java.io.ObjectOutputStream;
/*
序列化对象
1、NotSerializableException Student对象不支持序列化,需要实现Serializable接口
2、使用序列化的对象,必须实现Serializable接口
3、通过查看源代码发现,Serializable接口只是一个标志接口
接口中什么代码都没有。那么它起到什么作用呢?
起到标志作用,java虚拟机看到这个类实现了这个接口,就会给这个类特殊服务。
Java虚拟机看到这个接口之后,会给这个类生成一个序列化版本号。
4、序列化版本号有什么用?
java.io.InvalidClassException:
com.io.demo04.Student;
local class incompatible:
stream classdesc serialVersionUID = -5797858437193336959, (十年后)
local class serialVersionUID = 2310767740309256318 (十年前)
Java语言中是采用什么机制区分“类”的?
第一:首先通过类名进行比对,如果类名不一样,那肯定不是同一个类。
第二:如果类名一样,还能怎么区分类?靠序列化版本号进行区分。
以下场景:
小张编写了一个类:com.io.demo04.Student implements Serializable
小李编写了一个类:com.io.demo04.Student implements Serializable
不同的人编写了同一个类,但“这两个类确实不是同一个类”,这个时候序列化版本号就起作用了。
对于Java虚拟机来说,Java虚拟机是可以区分开这两个类的,因为这两个类都实现了Serializable接口,
都有默认的序列化版本号,它们的序列化版本号不一样,所以Java虚拟机能区分开。(这是自动生成序列化版本号的好处)
这种自动生成序列化版本号有什么缺陷呢?
缺点:一旦代码确定之后,就不能进行后续的修改,因为只要修改,必然会重新编译,
此时会生成全新的序列化版本号,这个时候Java虚拟机会认为这是一个全新的类。(这就麻烦了!)
最终结论:
凡是一个类实现了Serializable接口,建议给这个类提供一个固定不变的序列化版本号。
这样,以后这个类的代码即使修改了,但是版本号不变,Java虚拟机还是会认为是同一个类。
*/
public class Demo01 {
public static void main(String[] args) throws Exception {
//创建Student对象
Student student = new Student("娜可露露", 1001);
//创建序列化输出流
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("基础语法\src\com\io\students"));
//系列化对象
oos.writeObject(student);
//写完以后,要刷新管道
oos.flush();
//关闭最外层流
oos.close();
}
}
Student类实现Serializable接口:
package com.io.demo04;
import java.io.Serializable;
public class Student implements Serializable {
//Java虚拟机看到Serializable接口之后,会自动生成一个序列化版本号
//这里没有手动写出来,java虚拟机会默认提供这个序列化版本号
//建议序列化版本号手动写出来,不建议自动生成!
//Java虚拟机识别一个类的时候先通过类名,如果类名一致,再通过序列化版本号
private static final long serialVersionUID = 86834523123243189L; //指定序列化版本号
private String name;
private int id;
private String eMail; //后期改动的代码
//过了很久,Student这个类的源代码改动了。
//源代码改动之后,需要重新编译,编译之后生成全新的字节码文件。
//并且class文件再次运行的时候,Java虚拟机生成的序列化版本号也会发生相应的改变。
private int age; //后期改动的代码
public Student() {
}
public Student(String name, int id) {
this.name = name;
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", id=" + id +
'}';
}
}
反序列化:
package com.io.demo04;
import java.io.FileInputStream;
import java.io.ObjectInputStream;
/*
反序列化 ObjectInputStream
*/
public class Demo02 {
public static void main(String[] args) throws Exception {
//创建序列化输入流对象
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("基础语法\src\com\io\students"));
//反序列化,从文件读对象
Object obj = ois.readObject();
//反序列化回来的是一个学生对象,所以会调用学生对象的toString方法
System.out.println(obj);
//关闭最外层流
ois.close();
}
}
序列化多个对象
序列化多个对象就是用集合把多个对象装在一起,然后只需序列化集合。
package com.io.demo04;
import java.io.FileOutputStream;
import java.io.ObjectOutputStream;
import java.util.ArrayList;
import java.util.List;
/*
如何一次序列化多个对象?
可以将对象放到集合中,序列化集合
注意:参与序列化的ArrayList集合以及集合中的元素User都需要实现java.io.Serializable接口
*/
public class Demo03 {
public static void main(String[] args) throws Exception {
List<User> userList = new ArrayList<>();
userList.add(new User("迪丽热巴", 1001));
userList.add(new User("古力娜扎", 1002));
userList.add(new User("玛尔扎哈", 1003));
userList.add(new User("不知火舞", 1004));
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("基础语法\src\com\io\users"));
//写操作,序列化一个集合,这个集合对象中存放了很多其他对象
oos.writeObject(userList);
//刷新管道
oos.flush();
//关闭最外层流
oos.close();
}
}
User类实现Serializable接口:
被transient修饰的属性不会被序列化!
package com.io.demo04;
import java.io.Serializable;
public class User implements Serializable {
private String name;
private int id;
//transient关键字表示这个属性不会被序列化
// private transient int age; //age不参与序列化操作
public User() {
}
public User(String name, int id) {
this.name = name;
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", id=" + id +
'}';
}
}
反序列化集合:
需要判断读出来对象的类型与集合类型是否相同!
package com.io.demo04;
import java.io.FileInputStream;
import java.io.ObjectInputStream;
import java.util.List;
/*
反序列化集合
*/
public class Demo04 {
public static void main(String[] args) throws Exception {
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("基础语法\src\com\io\users"));
//反序列化对象,从文件中读入对象
// Object obj = ois.readObject();
// System.out.println(obj instanceof List);
// List<User> userList = obj;
List<User> userList = (List<User>) ois.readObject();
for (User user : userList) {
System.out.println(user.toString());
// System.out.println(user);
}
//关闭最外层流
ois.close();
}
}
配置文件
io流+Properties集合的联合使用
一个非常好的设计理念:
经常改变的数据,可以单独写到一个文件中,使用程序动态读取,以后只需要修改这个文件的内容,Java代码不需要改动,不需要重新编译,服务器也不需要重新部署,就可以拿到动态的信息。
package com.io.demo05;
import java.io.FileReader;
import java.util.Properties;
/*
io流+Properties集合的联合使用
一个非常好的设计理念:
经常改变的数据,可以单独写到一个文件中,使用程序动态读取,以后只需要修改
这个文件的内容,Java代码不需要改动,不需要重新编译,服务器也不需要重新部署,
就可以拿到动态的信息。
类似以上机制的这种文件叫做配置文件。
配置文件中内容的格式为:
key1 = value
key2 = value
的时候,我们把这种配置文件叫做属性配置文件。
java规范建议属性配置文件以properties结尾
Properties是专门存放属性配置文件内容的一个类。
*/
public class Demo01 {
public static void main(String[] args) throws Exception {
/*
Properties是一个Map集合,key和value都是String类型
想将userinfo文件中的数据加载到Properties对象中。
*/
//新建一个字符输入流对象
FileReader reader = new FileReader("基础语法\\src\\com\\io\\userinfo.properties");
//新建一个Map集合
Properties pro = new Properties();
//调用Properties对象中的load方法将文件中的数据加载到Map集合中
pro.load(reader); //文件中的数据顺着管道加载到Map集合中,其中左边作key,右边作value
//通过key获取value
String username = pro.getProperty("username");
System.out.println(username);
String password = pro.getProperty("password");
System.out.println(password);
}
}
属性配置文件:
username=admin
password=123456
###############在配置文件中#是注释############
#属性配置文件的key重复的话,value会自动覆盖!
#等号左右不要有空格
#password=234567