文件在程序中是以流的形式来操作的
流:数据在数据源(文件)和程序(内存)之间经历的路径
输入流:数据从数据源(文件)到程序(内存)的路径
输出流:数据从程序(内存)到数据源(文件)的路径
>相关方法
new File(String pathname)//根据路径构建一个File对象
new File(File parent,String child)//根据父目录文件+子路径构建
new File(String parent,String child)//根据父目录+子路径构建
createNewFile 创建新文件
import org.junit.jupiter.api.Test;
import java.io.File;
import java.io.IOException;
//演示创建文件
public class FileCreate {
public static void main(String[] args) {
}
//1.new File(String pathname)//根据路径构建一个File对象
@Test
public void create01(){
String filePath = "d:\news1.txt";
File file = new File(filePath);
try {
file.createNewFile();
System.out.println("文件创建成功");
} catch (IOException e) {
e.printStackTrace();
}
}
//2.new File(File parent,String child)//根据父目录文件+子路径构建
//假设路径为:d:\news2.txt
@Test
public void create02(){
File parentfile = new File("d:\");
String filename = "news2.txt";
//这里的file对象,在java程序中,只是一个对象而已
//只有执行了createNewFile方法,才会真正的在磁盘创建该文件
File file = new File(parentfile,filename);
try {
file.createNewFile();
System.out.println("文件创建成功");
} catch (IOException e) {
e.printStackTrace();
}
}
//3.new File(String parent,String child)//根据父目录+子路径构建
@Test
public void create03(){
String parentPath = "d:\";
//String parentPath ="d:/"; //也可以,但不推荐
String fileName = "news3.txt";
File file = new File(parentPath, fileName);
try {
file.createNewFile();
System.out.println("文件创建成功");
} catch (IOException e) {
e.printStackTrace();
}
}
}
>获取文件的相关信息
import org.junit.jupiter.api.Test;
import java.io.File;
public class FileInfo {
public static void main(String[] args) {
}
//获取文件信息
@Test
public void info(){
//先创建文件对象
File file = new File("d:\news1.txt");
//调用相应的方法,得到对应信息
System.out.println("文件名字="+file.getName());
System.out.println("文件是否可读="+file.canRead());
System.out.println("文件是否可写="+file.canWrite());
System.out.println("文件绝对路径="+file.getAbsolutePath());
System.out.println("文件父级目录="+file.getParent());
System.out.println("文件大小(单位:字节)="+file.length());
System.out.println("文件是否存在="+file.exists());
System.out.println("是不是一个文件="+file.isFile());
System.out.println("是不是一个目录="+file.isDirectory());
System.out.println("是不是隐藏文件="+file.isHidden());
//时间是从1970年午夜至文件最后修改时刻的毫秒数
System.out.println("文件最后修改时间="+file.lastModified());
}
}
>目录的操作和文件删除
mkdir创建一级目录、mkdirs创建多级目录、delete删除空目录或文件
import org.junit.jupiter.api.Test;
import java.io.File;
public class Directory {
public static void main(String[] args) {
}
//判断 d:\news1.txt是否存在,若存在,就删除
@Test
public void m1(){
String filePath = "d:\news1.txt";
File file = new File(filePath);
if(file.exists()){
if(file.delete()){
System.out.println(filePath+"删除成功");
}else{
System.out.println(filePath+"删除失败");
}
}else{
System.out.println("该文件不存在");
}
}
//判断 D:\demo02 是否存在,存在就删除,否则提示不存在
//从这里可以看出,在java编程中,目录也被当做文件
@Test
public void m2(){
String filePath = "d:\demo02";
File file = new File(filePath);
if(file.exists()){
if(file.delete()){
System.out.println(filePath+"删除成功");
}else{
System.out.println(filePath+"删除失败");
}
}else{
System.out.println("该目录不存在");
}
}
//判断 d:\demo\a\b\c 目录是否存在,若存在就提示已存在,否则就创建
@Test
public void m3(){
String directoryPath = "d:\demo\a\b\c";
File file = new File(directoryPath);
if(file.exists()){
System.out.println(directoryPath+"已存在");
}else {
if (file.mkdirs()) {
System.out.println(directoryPath + "创建成功");
} else {
System.out.println(directoryPath + "创建失败");
}
}
}
}
需要注意的是当删除某一目录时,必须保证该目录下没有其他文件才能正确删除,否则将删除失败。
import java.io.File;
public class IDL {
public static void main(String args[]) {
// 这里修改为自己的测试目录
File folder = new File("/tmp/java/");
deleteFolder(folder);
}
// 删除文件及目录
public static void deleteFolder(File folder) {
File[] files = folder.listFiles();
if (files != null) {
for (File f : files) {
if (f.isDirectory()) { //是目录
deleteFolder(f);
} else {
f.delete();
}
}
}
folder.delete();
}
}
DeleteFolder 方法不区分文件夹是否包含了内容。
即删除指定的文件夹时不考虑其中是否有内容。
一个目录其实就是一个 File 对象,它包含其他文件和文件夹。
如果创建一个 File 对象并且它是一个目录,那么调用 isDirectory() 方法会返回 true。
可以通过调用该对象上的 list() 方法,来提取它包含的文件和文件夹的列表。
1) String[] list() 用字符串形式返回目录下的全部文件
2) File[] listFiles() 用File对象形式返回目录下的全部文件
3) String[] list(FilenameFilter obj) 用字符串形式返回目录下指定类型的所有文件
4) File[] listFiles(FilenameFilter obj) 用File对象形式返回目录下指定类型的所有文件
指定类型的文件,即如:java、txt等扩展名的文件
String[] list(FilenameFilter obj) 用字符串形式返回目录下指定类型的所有文件
File[] listFiles(FilenameFilter obj) 用File对象形式返回目录下指定类型的所有文件这两个方法的参数FilenameFilter是一个接口,该接口有一个方法:
public boolean accept(File dir,String name);
File 对象 dirFile 调用 list 方法时,需向该方法传递一个实现FilenameFilter 接口的对象,list方法执行时,参数 obj 不断回调接口方法 accept(File dir,String name),该方法中的参数 dir为调用 list 的当前目录 dirFile,参数 name 被实例化为 dirFile 目录中的一个文件名,当接口方法返回 true 时,list 方法就将名字为name 的文件存放到返回的数组中.
//列出当前目录(应用程序所在的目录)下全部.java文件的名字
import java.io.*;
public class IIN {
public static void main(String args[]) {
File dirFile = new File(".");
FileAccept fileAccept = new FileAccept();
fileAccept.setExtendName("java");
String fileName[] = dirFile.list(fileAccept);
for (String name : fileName) {
System.out.println(name);
}
}
}
class FileAccept implements FilenameFilter {
private String extendName;
public void setExtendName(String s) {
extendName = "." + s;
}
public boolean accept(File dir, String name) { //重写接口中的方法
return name.endsWith(extendName);
}
}
>IO流原理及流的分类 >IO流原理扩展:
endsWith()方法用于判断字符串是否以指定后缀结尾,返回True/False
1.I/O是Input/Output的缩写,I/O技术是非常实用的技术,用于处理数据传输。如读/写文件,网络通讯等。
2.Java程序中,对于数据的输入/输出操作以”流(stream)”的方式进行。
3.java.io包下提供了各种“流”类和接口,用以获取不同种类的数据,并通过方法输入或输出数据。
4.输入input:读取外部数据(磁盘、光盘等存储设备的数据)到程序(内存)中。
5.输出output:将程序(内存)数据输出到磁盘、光盘等存储设备中。
>按操作数据单位不同分为:字节流(8 bit),字符流(按字符,字符对应字节要看编码)
>按数据流的流向不同分为:输入流,输出流
>按流的角色的不同分为:节点流,处理流/包装流
1)Java的IO流共涉及40多个类,实际上非常规则,都是从如上4个抽象基类派生的。
2)由这四个类派生出来的子类名称都是以其父类名作为子类名后缀。
>InputStream:字节输入流
>InputStream抽象类是所有类字节输入流的超类
>InputStream常用的子类
1.FilelnputStream:文件输入流
2.BufferedInputStream:缓冲字节输入流
3.ObjectInputStream:对象字节输入流
1)FilelnputStream:文件输入流
import org.junit.jupiter.api.Test;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
//演示FileInputStream的使用(字节输入流 文件->程序)
//例:使用FileInputStream读取hello.txt文件,并将文件内容显示到控制台
public class FileInputStream1 {
public static void main(String[] args) {
}
@Test
public void readFile01(){
String filePath = "d:\hello.txt";
int readData = 0;
byte[] buf = new byte[8]; //字节数组,一次读取8个字节
int readLen = 0;
FileInputStream fileInputStream = null;
try {
//创建了 FileInputStream对象,用于读取文件
//不可用FileInputStream fileInputStream = new FileInputStream(filePath);
//因为对象在try中定义的话,作用域只在try,finally处无法使用
fileInputStream = new FileInputStream(filePath);
//read():从该输入流读取一个字节的数据
//从该输入流读取一个字节的数据。 如果没有输入可用,此方法将阻止。
//结果:数据的下一个字节,如果达到文件的末尾, -1 。
// while((readData = fileInputStream.read()) != -1){
// System.out.print((char)readData); //转成char显示
// }
//read(byte[] b):从该输入流读取最多 b.length个字节的数据为字节数组。
//如果读取正常,返回实际读取的字节数 返回-1,则表示读取完毕
while((readLen = fileInputStream.read(buf)) != -1){
System.out.print(new String(buf,0, readLen)); //显示
}
} catch (IOException e) {
e.printStackTrace();
} finally {
//关闭文件流,释放资源
try {
fileInputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
2)FileOutputStream:文件输出流
import org.junit.jupiter.api.Test;
import java.io.FileOutputStream;
import java.io.IOException;
//例:使用FileOutputStream 在a.txt文件中写入“hello,world"
//如果文件不存在,会创建文件(注意:前提是目录已存在)
public class FileOutputStream01 {
public static void main(String[] args) {
}
@Test
//演示使用FileOutputStream将数据写到文件中
public void writeFile(){
//创建 FileOutputStream 对象
String filePath = "d:\a.txt";
FileOutputStream fileOutputStream = null;
try {
//得到FileOutputStream对象
//1.new FileOutputStream(filePath)创建方式,当写入内容时,会覆盖原内容
//2.new FileOutputStream(filePath)创建方式,当写入内容时,是追加到文件后
fileOutputStream = new FileOutputStream(filePath);
// //写入一个字节 write(int b)
// fileOutputStream.write('h');
//写入字符串 write(byte[] b)
String str = "hello,world";
//str.getBytes() 即把 字符串->字节数组
fileOutputStream.write(str.getBytes());
//write(byte[] b,int off,int len)
//将 len字节从位于偏移量 off的指定字节数组写入此文件输出流。
//fileOutputStream.write(str.getBytes(),0,str.length());
} catch (IOException e) {
e.printStackTrace();
}finally {
try {
fileOutputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
2>FileReader/FileWriter 文件字符输入/输出流//1.new FileOutputStream(filePath)创建方式,当写入内容时,会覆盖原内容
//2.new FileOutputStream(filePath)创建方式,当写入内容时,是追加到文件后
1)FileReader 文件字符输入流
>FileReader相关方法:
1)new FileReader(File/String)
2)read:每次读取单个字符,返回该字符,如果到文件末尾返回-1
3)read(char[]):批量读取多个字符到数组,返回读取到的字符数, 如果到文件末尾返回-1
相关API:
1)new String(char[]):将char[]转换成String
2) new String(char[],off,len):将char[]的指定部分转换成String
import java.io.FileReader;
import java.io.IOException;
public class FileReader01 {
public static void main(String[] args) {
int data = 0;
int readLen = 0;
char[] buf = new char[8];
//1.创建FileReader对象
FileReader fileReader = null;
try {
fileReader = new FileReader("d:\story.txt");
//循环读取 使用read
// while((data = fileReader.read())!= -1){
// System.out.print((char)data);
// }
while((readLen = fileReader.read(buf))!= -1){
System.out.print(new String(buf,0,readLen));
}
} catch (IOException e) {
e.printStackTrace();
}finally {
try {
fileReader.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
2)FileWriter 文件字符输出流
>FileWriter常用方法
1 new FileWriter(File/String):覆盖模式,相当于流的指针在首端
2) new FileWriter(File/String,true):追加模式,相当于流的指针在尾端
3) write(int):写入单个字符
4)write(char[]):写入指定数组
5) write(char[],off,len):写入指定数组的指定部分
6) write(string):写入整个字符串
7) write(string,off,len):写入字符串的指定部分
相关API:
String类:toCharArray:将String转换成char[]
注意:
FileWriter使用后,必须要关闭(close)或刷新(flush),否则写入不到指定的文件!
import java.io.FileWriter;
import java.io.IOException;
public class FileWrite1 {
public static void main(String[] args) {
char[] chars = {'a','b','c'};
FileWriter fileWriter = null;
try {
fileWriter = new FileWriter("d:\note.txt");
//3) write(int):写入单个字符
fileWriter.write('H');
//4)write(char[]):写入指定数组
fileWriter.write(chars);
//5) write(char[],off,len):写入指定数组的指定部分
fileWriter.write("仪宝下午好".toCharArray(),0,3);
//6) write(string):写入整个字符串
fileWriter.write("下午好,宝");
//7) write(string,off,len):写入字符串的指定部分
fileWriter.write("上海天津",0,2);
//在数据量大的情况下, 可使用循环操作
} catch (IOException e) {
e.printStackTrace();
}finally {
//FileWriter使用后,必须要关闭(close)或刷新(flush),否则写入不到指定的文件!
try {
fileWriter.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
>节点流和处理流
1.节点流可以从一个特定的数据源读写数据, 如FileReader、FileWriter
2.处理流(也叫包装流)是“连接”在已存在的流(节点流或处理流)之上,为程序提供更为强大的读写功能,如BufferedReader、BufferedWriter
>节点流和处理流的区别和联系
1.节点流是底层流/低级流,直接跟数据源相接。
2.处理流包装节点流,既可以消除不同节点流的实现差异,也可以提供更方便的方法来完成输入输出。
3.处理流(也叫包装流)对节点流进行包装,使用了修饰器设计模式,不会直接与数据源相连。
>处理流的功能主要体现在以下两个方面:
1.性能的提高:主要以增加缓冲的方式来提高输入输出的效率。
2.操作的便捷:处理流可能提供了一系列便捷的方法来一次输入输出大批量的数据,使用更加灵活方便
BufferedReader和BufferedWriter属于字符流,是按照字符来读取数据的!不要去操作二进制文件【声音、视频等】,否则可能造成文件损坏。
关闭时,只需要关闭外层流即可
1)BufferedReader 缓冲输入流
import java.io.BufferedReader;
import java.io.FileReader;
public class BufferedReader_ {
public static void main(String[] args) throws Exception {
String filePath = "d:\a.java";
//创建bufferedReader
BufferedReader bufferedReader = new BufferedReader(new FileReader(filePath));
//读取
String line; //按行读取,效率高
//1.bufferedReader.readLine() 是按行读取文件
//2.当返回null时,表示文件读取完毕
//3.readLine 读取一行内容 但是不提供换行!
while ((line = bufferedReader.readLine()) != null) {
System.out.println(line);
}
//关闭流,只需关闭BufferedReader,因为底层会自动的去关闭节点流
bufferedReader.close();
}
}
1)BufferedWriter 缓冲输出流
import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;
public class BufferedWriter_ {
public static void main(String[] args) {
String filePath = "d:\OK.txt";
BufferedWriter bufferedWriter = null;
//创建一个BufferedWriter对象
try {
bufferedWriter = new BufferedWriter(new FileWriter(filePath));
bufferedWriter.write("hello,仪宝下午好");
bufferedWriter.newline();//插入一个和系统相关的换行
bufferedWriter.write("hello,仪宝下午好");
} catch (IOException e) {
e.printStackTrace();
}finally {
try {
bufferedWriter.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
4 >BufferedInputStream/BufferedOutputStream
1)BufferedInputStream
2)BufferedOutputStream
4>ObjectInputStreamObjectOutputStream 对象流
>看一个需求
1.将int num=100这个int数据保存到文件中,注意不是100数字,而是int100,并且,能够
从文件中直接恢复int 100
2.将Dog dog=new Dog(“小黄”3)这个dog对象保存到文件中,并且能够从文件恢复.
3上面的要求,就是能够将基本数据类型或者对象 进行 序列化 和 反序列化操作
>序列化和反序列化
序列化就是在保存数据时,保存数据的值和数据类型
2.反序列化就是在恢复数据时,恢复数据的值和数据类型
3.需要让某个对象支持序列化机制,则必须让其类是可序列化的,为了让某个类是可序列化的,该类必须实现如下两个接口之一:
>Serializable//这是一个标记接口
>Externalizable//该接口有方法需要实现,因此我们一般实现上面的 Serializable接口
1)ObjectInputStream 对象输入流
2)ObjectOutputStream 对象输出流



