ThinkChat🤖让你学习和工作更高效,注册即送10W Token,即刻开启你的AI之旅 广告
## IO 大多数应用程序都需要实现与设备之间的数据传输,例如键盘可以输入数据,显示器可以显示程序的运行结果等。在Java中,将这种通过不同输入输出设备(键盘,内存,显示器,网络等)之间的数据传输抽象表述为“流”,程序允许通过流的方式与输入输出设备进行数据传输。Java中的“流”都位于java.io包中,称为IO(输入输出)流。 IO流有很多种,按照操作数据的不同,可以分为字节流和字符流,按照数据传输方向的不同又可分为输入流和输出流,程序从输入流中读取数据,向输出流中写入数据。在IO包中,字节流的输入输出流分别用java.io.InputStream 和java.io.OutputStream 表示,字符流的输入输出流分别用java.io.Reader和java.io.Writer表示,具体分类如图所示。 ![](http://47.107.171.232/easily-j/images/20190111/9766628d-b101-45b0-b592-9af06cde7531.png) ## 字节流 #### 字节流概念 在计算机中,无论是文本、图片、音频还是视频,所有的文件都是以二进制(字节)形式存在,IO流中针对字节的输入输出提供了一系列的流,统称为字节流。字节流是程序中最常用的流,根据数据的传输方向可将其分为字节输入流和字节输出流。在JDK 中,提供了两个抽象类InputStream 和OutputStream,它们是字节流的顶级父类,所有的字节输入流都继承自InputStream,所有的字节输出流都继承自OutputStream。为了方便理解,可以把InputStream 和OutputStream 比作两根“水管”,如图所示。 ![](http://47.107.171.232/easily-j/images/20190111/bcfbd193-8f57-444c-a6f4-d8e0d7ac28ea.png) 图中,InputStream 被看成一个输入管道,OutputStream 被看成一个输出管道,数据通过InputStream 从源设备输入到程序,通过OutputStream 从程序输出到目标设备,从而实现数据的传输。由此可见,IO流中的输入输出都是相对于程序而言的。 在JDK中,InputStream 和OutputStream 提供了一系列与读写数据相关的方法,接下来先来了解一下InputStream 的常用方法,如表所示。 ![](http://47.107.171.232/easily-j/images/20190111/961a30d6-deb2-4a38-bea3-2c417c2b222f.png) 表中列举了InputStream 的四个常用方法。前三个read()方法都是用来读数据的,其中,第一个read()方法是从输入流中逐个读入字节,而第二个和第三个read()方法则将若干字节以字节数组的形式一次性读入,从而提高读数据的效率。在进行IO 流操作时,当前IO流会占用一定的内存,由于系统资源宝贵,因此,在IO 操作结束后,应该调用close()方法关闭流,从而释放当前IO流所占的系统资源。 与InputStream 对应的是OutputStream。OutputStream 是用于写数据的,因此OutputStream 提供了一些与写数据有关的方法,如表所示。 ![](http://47.107.171.232/easily-j/images/20190111/d2835b5e-42af-4e74-8869-ae9775214172.png) 表中,列举了OutputStream 类的五个常用方法。前三个是重载的write()方法,都是用于向输出流写入字节,其中,第一个方法逐个写入字节,后两个方法是将若干个字节以字节数组的形式一次性写入,从而提高写数据的效率。flush()方法用来将当前输出流缓冲区(通常是字节数组)中的数据强制写入目标设备,此过程称为刷新。close()方法是用来关闭流并释放与当前IO流相关的系统资源。 InputStream 和OutputStream 这两个类虽然提供了一系列和读写数据有关的方法,但是这两个类是抽象类,不能被实例化,因此,针对不同的功能,InputStream 和OutputStream 提供了不同的子类,这些子类形成了一个体系结构,如图所示。 ![](http://47.107.171.232/easily-j/images/20190111/dd8d76df-53c9-480b-a7f2-b75ee2db7c7c.png) ![](http://47.107.171.232/easily-j/images/20190111/baea8226-fb97-432b-8b3f-d970844abd75.png) 从图中可以看出,InputStream 和OutputStream 的子类有很多是大致对应的,比如ByteArrayInputStream 和ByteArrayOutputStream,FileInputStream 和FileOutputStream 等。图中所列出的IO 流都是程序中很常见的,接下来将逐步为大家讲解这些流的具体用法。 #### 字节流读写文件 由于计算机中的数据基本都保存在硬盘的文件中,因此操作文件中的数据是一种很常见的操作。在操作文件时,最常见的就是从文件中读取数据并将数据写入文件,即文件的读写。针对文件的读写,JDK 专门提供了两个类,分别是FileInputStream 和FileOutputStream。 FileInputStream 是InputStream 的子类,它是操作文件的字节输入流,专门用于读取文件中的数据。由于从文件读取数据是重复的操作,因此需要通过循环语句来实现数据的持续读取。接下来通过一个案例来实现字节流对文件数据的读取,首先在D盘目录下创建一个文本文件IO.txt,在文件中输入内容“小海绵”,具体代码如例所示。 ```java import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.IOException; public class IOTest { public static void main(String[] args) { try { // 创建一个文件字节输入流 FileInputStream in = new FileInputStream("D:/IO.txt"); int b = 0; // 定义一个int 类型的变量b,记住每次读取的一个字节 while (true) { b = in.read(); // 变量b 记住读取的一个字节 if (b == -1) { // 如果读取的字节为-1,跳出while 循环 break; } System.out.println(b); // 否则将b 写出 } in.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 运行结果: ``` 208 161 186 163 195 224 ``` 例中,创建的字节流FileInputStream 通过read()方法将当前目录文件“D://IO.txt”中的数据读取并打印。通常情况下读取文件应该输出字符,之所以输出数字是因为硬盘上的文件是以字节的形式存在的,在“IO.txt”文件中,字符'小','海','绵'各占2个字节,因此,最终结果显示的就是文件中的六个字节所对应的十进制数。 与FileInputStream对应的是FileOutputStream。FileOutputStream 是OutputStream 的子类,它是操作文件的字节输出流,专门用于把数据写入文件。接下来通过一个案例来演示如何将数据写入文件,如例所示。 ```java import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; public class IOTest { public static void main(String[] args) { try { // 创建一个文件字节输出流 FileOutputStream out = new FileOutputStream("D:/example.txt"); String str = "小海绵"; byte[] b = str.getBytes(); for (int i = 0; i < b.length; i++) { out.write(b[i]); } out.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 程序运行后,会在D盘目录下生成一个新的文本文件example.txt,打开此文件,文件内容为小海绵。 通过运行结果可以看出,通过FileOutputStream 写数据时,自动创建了文件example.txt,并将数据写入文件。需要注意的是,如果是通过FileOutputStream 向一个已经存在的文件中写入数据,那么该文件中的数据首先会被清空,再写入新的数据。若希望在已存在的文件内容之后追加新内容,则可使用FileOutputStream 的构造函数FileOutputStream(StringfileName,booleanappend)来创建文件输出流对象,并把append参数的值设置为true。接下来通过一个案例来演示如何将数据追加到文件末尾,如例所示。 ```java import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.io.OutputStream; public class IOTest { public static void main(String[] args) { try { OutputStream out = new FileOutputStream("D:/example.txt", true); String str = "炒鸡帅"; byte[] b = str.getBytes(); for (int i = 0; i < b.length; i++) { out.write(b[i]); } out.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 查看D://example.txt文件,文件中的内容为 小海绵炒鸡帅。 #### 文件的拷贝 在应用程序中,IO流通常都是成对出现的,即输入流和输出流一起使用。例如文件的拷贝就需要通过输入流来读取文件中的数据,通过输出流将数据写入文件。接下来通过一个案例来演示如何进行文件内容的拷贝,首先在D盘创建文件夹one,和two,然后在one文件夹中存放一个“example.txt”文件,拷贝文件的代码如例所示。 ```java import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; public class IOTest { public static void main(String[] args) { try { // 创建一个字节输入流,用于读取当前目录下source 文件夹中的mp3 文件 InputStream in = new FileInputStream("D:/one/example.txt"); // 创建一个文件字节输出流,用于将读取的数据写入target 目录下的文件中 OutputStream out = new FileOutputStream("D:/two/example.txt"); int len; // 定义一个int 类型的变量len,记住每次读取的一个字节 long begintime = System.currentTimeMillis(); // 获取拷贝文件前的系统时间 while ((len = in.read()) != -1) { // 读取一个字节并判断是否读到文件末尾 out.write(len); // 将读到的字节写入文件 } long endtime = System.currentTimeMillis(); // 获取文件拷贝结束时的系统时间 System.out.println("拷贝文件所消耗的时间是: " + (endtime - begintime) + "毫秒"); in.close(); out.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 运行结果: ``` 拷贝文件所消耗的时间是: 2毫秒 ``` 在拷贝过程中,通过while循环将字节逐个进行拷贝。每循环一次,就通过FileInputStream 的read()方法读取一个字节,并通过FileOutputStream 的write()方法将该字节写入指定文件,循环往复,直到len的值为-1,表示读取到了文件的末尾,结束循环,完成文件的拷贝。程序运行结束后,会在命令行窗口打印拷贝文件所消耗的时间。 #### 字节流的缓冲区 虽然上一个案例实现了文件的拷贝,但是一个字节一个字节的读写,需要频繁的操作文件,效率非常低,这就好比从北京运送烤鸭到上海,如果有一万只烤鸭,每次运送一只,就必须运输一万次,这样的效率显然非常低。为了减少运输次数,可以先把一批烤鸭装在车厢中,这样就可以成批的运送烤鸭,这时的车厢就相当于一个临时缓冲区。当通过流的方式拷贝文件时,为了提高效率也可以定义一个字节数组作为缓冲区。在拷贝文件时,可以一次性读取多个字节的数据,并保存在字节数组中,然后将字节数组中的数据一次性写入文件。接下来通过一个案例来学习如何使用缓冲区拷贝文件,如例所示。 ```java import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; public class IOTest { public static void main(String[] args) { try { // 创建一个字节输入流,用于读取当前目录下source 文件夹中的mp3 文件 InputStream in = new FileInputStream("D:/one/example.txt"); // 创建一个文件字节输出流,用于将读取的数据写入当前目录的target 文件中 OutputStream out = new FileOutputStream("D:/one/example.txt"); // 以下是用缓冲区读写文件 byte[] buff = new byte[1024]; // 定义一个字节数组,作为缓冲区 // 定义一个int 类型的变量len 记住读取读入缓冲区的字节数 int len; long begintime = System.currentTimeMillis(); while ((len = in.read(buff)) != -1) { // 判断是否读到文件末尾 out.write(buff, 0, len); // 从第一个字节开始,向文件写入len 个字节 } long endtime = System.currentTimeMillis(); System.out.println("拷贝文件所消耗的时间是: " + (endtime - begintime) + "毫秒"); in.close(); out.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 运行结果: ``` 拷贝文件所消耗的时间是: 0毫秒 ``` 在拷贝过程中,使用while循环语句逐渐实现字节文件的拷贝,每循环一次,就从文件读取若干字节填充字节数组,并通过变量len记住读入数组的字节数,然后从数组的第一个字节开始,将len个字节依次写入文件。循环往复,当len值为-1时,说明已经读到了文件的末尾,循环会结束,整个拷贝过程也就结束了,最终程序将整个拷贝过程所消耗的时间打印了出来。 通过两种拷贝方式的对比,可以看出拷贝文件所消耗的时间明显减少了,从而说明缓冲区读写文件可以有效的提高程序的效率。这是因为程序中的缓冲区就是一块内存,用于存放暂时输入输出的数据,使用缓冲区减少了对文件的操作次数,所以可以提高读写数据的效率。 #### 装饰设计模式 俗话说“人靠衣装马靠鞍”,漂亮得体的装扮不仅能提升形象,还能提高竞争力。在程序设计中,同样可以通过“装饰”一个类,增强它的功能。装饰设计模式就是通过包装一个类,动态地为它增加功能的一种设计模式。 装饰设计模式在现实生活中随处可见,比如买了一辆车,想为新车装一个倒车雷达,这就相当于为这辆汽车增加新的功能。接下来通过一个案例来实现上述过程,如例所示。 ```java class Car { private String carName; // 定义一个属性,代表车名 public Car(String carName) { this.carName = carName; } public void show() { // 实现Car 的show()方法 System.out.println("我是" + carName + ",具有基本功能"); } } // 定义一个类RadarCar class RadarCar { public Car myCar; public RadarCar(Car myCar) { // 通过构造方法接收被包装的对象 this.myCar = myCar; } public void show() { myCar.show(); System.out.println("具有倒车雷达功能"); // 实现功能的增强 } } public class IOTest { public static void main(String[] args) { Car benz = new Car("Benz"); // 创建一个NewCar 对象 System.out.println("--------------包装前--------------"); benz.show(); RadarCar decoratedCar_benz = new RadarCar(benz); // 创建一个RadarCar 对象 System.out.println("--------------包装后--------------"); decoratedCar_benz.show(); } } ``` 运行结果: ``` --------------包装前-------------- 我是Benz,具有基本功能 --------------包装后-------------- 我是Benz,具有基本功能 具有倒车雷达功能 ``` 例实现了RadarCar类对Car类的包装。包装类RadarCar的构造方法中接收一个Car类型的实例对象。通过运行结果可以看出,当RadarCar对象调用show()方法时,被RadarCar包装后的对象benz不仅具有车的基本功能,而且具有了倒车雷达的功能。 #### 字节缓冲流 在IO包中提供两个带缓冲的字节流,分别是BufferedInputStream和BufferdOutputStream,这两个流都使用了装饰设计模式。它们的构造方法中分别接收InputStream 和OutputStream 类型的参数作为被包装对象,在读写数据时提供缓冲功能。应用程序、缓冲流和底层字节流之间的关系如图所示。 ![](http://47.107.171.232/easily-j/images/20190111/dfccccb9-741b-419b-8e47-5ca52e4540f6.png) 从图中可以看出应用程序是通过缓冲流来完成数据读写的,而缓冲流又是通过底层被包装的字节流与设备进行关联的。接下来通过一个案例来学习BufferedInputStream和BufferedOutputStream 这两个流的用法,如例所示。 ```java import java.io.BufferedInputStream; import java.io.BufferedOutputStream; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; public class IOTest { public static void main(String[] args) { try { // 创建一个带缓冲区的输入流 BufferedInputStream bis = new BufferedInputStream(new FileInputStream("D:/example.txt")); // 创建一个带缓冲区的输出流 BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("D:/example.txt")); int len; while ((len = bis.read()) != -1) { bos.write(len); } bis.close(); bos.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 例中,创建了BufferedInputStream 和BufferedOutputStream 两个缓冲流对象,这两个流内部都定义了一个大小为8192的字节数组,当调用read()或者write()方法读写数据时,首先将读写的数据存入定义好的字节数组,然后将字节数组的数据一次性读写到文件中,这种方式与字节流的缓冲区类似,都对数据进行了缓冲,从而有效的提高了数据的读写效率。 ## 字符流 #### 字符流定义及基本用法 前面我们讲过InputStream 类和OutputStream 类在读写文件时操作的都是字节,如果希望在程序中操作字符,使用这两个类就不太方便,为此JDK 提供了字符流。同字节流一样,字符流也有两个抽象的顶级父类,分别是Reader和Writer。其中Reader是字符输入流,用于从某个源设备读取字符,Writer是字符输出流,用于向某个目标设备写入字符。Reader和Writer作为字符流的顶级父类,也有许多子类,接下来通过继承关系图来列出Reader和Writer的一些常用子类,如图所示。 ![](http://47.107.171.232/easily-j/images/20190111/c2d5f0fd-0f90-4bb8-922a-247e2f96539e.png) ![](http://47.107.171.232/easily-j/images/20190111/eb650e26-dc33-4eb3-8ad9-6860954b7e1e.png) 从图可以看到,字符流的继承关系与字节流的继承关系有些类似,很多子类都是成对(输入流和输出流)出现,其中FileReader和FileWriter用于读写文件,BufferedReader和BufferedWriter是具有缓冲功能的流,它们可以提高读写效率。 #### 字符流操作文件 在程序开发中,经常需要对文本文件的内容进行读取,如果想从文件中直接读取字符便可以使用字符输入流FileReader,通过此流可以从关联的文件中读取一个或一组字符。接下来首先在D盘目录下新建文件“example.txt”并在其中输入字符“小海绵”,然后通过一个案例来学习如何使用FileReader读取文件中的字符,如例所示。 ```java import java.io.FileNotFoundException; import java.io.FileReader; import java.io.IOException; public class IOTest { public static void main(String[] args) { try { // 创建一个FileReader 对象用来读取文件中的字符 FileReader reader = new FileReader("D:/example.txt"); int ch; // 定义一个变量用于记录读取的字符 while ((ch = reader.read()) != -1) { // 循环判断是否读取到文件的末尾 System.out.println((char) ch); // 不是字符流末尾就转为字符打印 } reader.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 运行结果: ``` 小 海 绵 ``` 例实现了读取文件字符的功能。首先创建一个FileReader对象与文件关联,然后通过while循环每次从文件中读取一个字符并打印,这样便实现了FileReader读文件字符的操作。需要注意的是,字符输入流的read()方法返回的是int类型的值,如果想获得字符就需要进行强制类型转换。 例讲解了如何使用FileReader读取文件中的字符,如果要向文件中写入字符就需要使用FileWriter类。FileWriter是Writer的一个子类,接下来通过一个案例来学习如何使用FileWriter将字符写入文件,如例所示。 ```java import java.io.FileNotFoundException; import java.io.FileWriter; import java.io.IOException; public class IOTest { public static void main(String[] args) { try { // 创建一个FileWriter 对象用于向文件中写入数据 FileWriter writer = new FileWriter("D:/example.txt", true); String str = "小哥哥"; writer.write(str); // 将字符数据写入到文本文件中 writer.write("\r\n"); // 将输出语句换行 writer.close(); // 关闭写入流,释放资源 } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 再次运行程序就可以实现在文件中追加内容的效果。 接下来通过一个案例来学习如何使用BufferedReader和BufferedWriter实现文件的拷贝,如例所示。 ```java import java.io.BufferedReader; import java.io.BufferedWriter; import java.io.FileNotFoundException; import java.io.FileReader; import java.io.FileWriter; import java.io.IOException; public class IOTest { public static void main(String[] args) { try { FileReader reader = new FileReader("D:/one/example.txt"); // 创建一个BufferedReader 缓冲对象 BufferedReader br = new BufferedReader(reader); FileWriter writer = new FileWriter("D:/one/example.txt"); // 创建一个BufferdWriter 缓冲对象 BufferedWriter bw = new BufferedWriter(writer); String str; while ((str = br.readLine()) != null) { // 每次读取一行文本,判断是否到文件末尾 bw.write(str); bw.newLine(); // 写入一个换行符,该方法会根据不同的操作系统生成相应的换行符 } br.close(); bw.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 在例中,首先对输入输出流进行了包装,并通过一个while循环实现了文本文件的拷贝。在拷贝过程中,每次循环都使用readLine()方法读取文件的一行,然后通过write()方法写入目标文件。其中readLine()方法会逐个读取字符,当读到回车\'r'或换行\' n'时会将读到的字符作为一行的内容返回。 需要注意的是,由于包装流内部使用了缓冲区,在循环中调用BufferedWriter的write()方法写字符时,这些字符首先会被写入缓冲区,当缓冲区写满时或调用close()方法时,缓冲区中的字符才会被写入目标文件。因此在循环结束时一定要调用close()方法,否则极有可能会导致部分存在缓冲区中的数据没有被写入目标文件。 #### 转换流 前面提到IO流可分为字节流和字符流,有时字节流和字符流之间也需要进行转换。在JDK中提供了两个类可以将字节流转换为字符流,它们分别是InputStreamReader和OutputStreamWriter。 转换流也是一种包装流,其中OutputStreamWriter是Writer的子类,它可以将一个字节输出流包装成字符输出流,方便直接写入字符,而InputStreamReader是Reader的子类,它可以将一个字节输入流包装成字符输入流,方便直接读取字符。通过转换流进行数据读写的过程如图所示。 ![](http://47.107.171.232/easily-j/images/20190111/e7663ad2-4ad5-4632-a1a5-48b0d2b3b90c.png) 接下来通过一个案例来学习如何将字节流转为字符流,为了提高读写效率,可以通过BufferedReader和BufferedWriter对转换流进行包装,具体代码如例所示。 ```java import java.io.BufferedReader; import java.io.BufferedWriter; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStreamReader; import java.io.OutputStreamWriter; public class IOTest { public static void main(String[] args) { try { FileInputStream in = new FileInputStream("D:/example.txt"); // 创建字节输入流 InputStreamReader isr = new InputStreamReader(in); // 将字节流输入转换成字符输入流 BufferedReader br = new BufferedReader(isr); // 对字符流对象进行包装 FileOutputStream out = new FileOutputStream("D:/example.txt"); // 将字节输出流转换成字符输出流 OutputStreamWriter osw = new OutputStreamWriter(out); BufferedWriter bw = new BufferedWriter(osw); // 对字符输出流对象进行包装 String line; while ((line = br.readLine()) != null) { // 判断是否读到文件末尾 bw.write(line); // 输出读取到的文件 } br.close(); bw.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 例实现了字节流和字符流之间的转换,将字节流转换为字符流,从而实现直接对字符的读写。需要注意的是,在使用转换流时,只能针对操作文本文件的字节流进行转换,如果字节流操作的是一张图片,此时转换为字符流就会造成数据丢失。 ## File类 本章前面讲解的IO流可以对文件的内容进行读写操作,在应用程序中还会经常对文件本身进行一些常规操作,例如创建一个文件,删除或者重命名某个文件,判断硬盘上某个文件是否存在,查询文件最后修改时间等。针对文件的这类操作,JDK 中提供了一个File类,该类封装了一个路径,并提供了一系列的方法用于操作该路径所指向的文件,接下来围绕File类展开详细讲解。 #### File类的常用方法 File类用于封装一个路径,这个路径可以是从系统盘符开始的绝对路径,如D:/example.txt,也可以是相对于当前目录而言的相对路径,如src/example.txt。File类内部封装的路径可以指向一个文件,也可以指向一个目录,在File类中提供了针对这些文件或目录的一些常规操作。接下来首先介绍一下File类常用的构造方法,如表所示。 ![](http://47.107.171.232/easily-j/images/20190111/6c73686e-16fc-4ed6-b676-390bb5a68b14.png) 表中列出了File类的三个构造方法。通常来讲,如果程序只处理一个目录或文件,并且知道该目录或文件的路径,使用第一个构造方法较方便。如果程序处理的是一个公共目录中的若干子目录或文件,那么使用第二个或者第三个构造方法会更方便。 File类中提供了一系列方法,用于操作其内部封装的路径指向的文件或者目录,例如判断文件/目录是否存在、创建、删除文件/目录等。接下来介绍一下File类中的常用方法,如表所示。 ![](http://47.107.171.232/easily-j/images/20190111/62735cf7-6e4a-41f1-a4cf-6c27f56dab7e.png) ![](http://47.107.171.232/easily-j/images/20190111/7e92b847-0b5a-4b18-a032-cac4fb58bff5.png) 表中,列出了File类的一系列常用方法,此表仅仅通过文字对File类的方法进行介绍,对于初学者来说很难弄清它们之间的区别,接下来,首先在D盘目录下创建一个文件“example.txt”并输入内容“小海绵”,然后通过一个案例来演示File类的常用方法,如例所示。 ```java import java.io.File; public class IOTest { public static void main(String[] args) { File file = new File("D:/example.txt"); // 创建File 文件对象,表示一个文件 // 获取文件名称 System.out.println("文件名称:" + file.getName()); // 获取文件的相对路径 System.out.println("文件的相对路径:" + file.getPath()); // 获取文件的绝对路径 System.out.println("文件的绝对路径:" + file.getAbsolutePath()); // 获取文件的父路径 System.out.println("文件的父路径:" + file.getParent()); // 判断文件是否可读 System.out.println(file.canRead() ? "文件可读" : "文件不可读"); // 判断文件是否可写 System.out.println(file.canWrite() ? "文件可写" : "文件不可写"); // 判断是否是一个文件 System.out.println(file.isFile() ? "是一个文件" : "不是一个文件"); // 判断是否是一个目录 System.out.println(file.isDirectory() ? "是一个目录" : "不是一个目录"); // 判断是否是一个绝对路径 System.out.println(file.isAbsolute() ? "是绝对路径" : "不是绝对路径"); // 得到文件最后修改时间 System.out.println("最后修改时间为:" + file.lastModified()); // 得到文件的大小 System.out.println("文件大小为:" + file.length() + " bytes"); // 是否成功删除文件 System.out.println("是否成功删除文件" + file.delete()); } } ``` 运行结果: ``` 文件名称:example.txt 文件的相对路径:D:\example.txt 文件的绝对路径:D:\example.txt 文件的父路径:D:\ 文件可读 文件可写 是一个文件 不是一个目录 是绝对路径 最后修改时间为:1547187548383 文件大小为:6 bytes 是否成功删除文件true ``` #### 遍历目录下的文件 在表中列举的方法中有一个list()方法,该方法用于遍历某个指定目录下的所有文件的名称,例8-25中没有演示该方法的使用,接下来通过一个案例来演示list()方法的用法,如例所示。 ```java import java.io.File; public class IOTest { public static void main(String[] args) { File file = new File("D:/"); // 创建File 对象 if (file.isDirectory()) { // 判断File 对象对应的目录是否存在 String[] names = file.list(); // 获得目录下的所有文件的文件名 for (String name : names) { System.out.println(name); // 输出文件名 } } } } ``` 例中,创建了一个File对象,封装了一个路径,通过调用File的isDirectory()方法判断路径指向的是否为存在的目录,如果存在就调用list()方法,获得一个String类型的数组names,数组中包含这个目录下所有文件的文件名。接着通过循环遍历数组names,依次打印出每个文件的文件名。 例实现了遍历一个目录下所有的文件,有时程序只是需要得到指定类型的文件,如获取指定目录下所有的“.java”文件。针对这种需求,File类中提供了一个重载的list(FilenameFilterfilter)方法,该方法接收一个FilenameFilter 类型的参数。FilenameFilter是一个接口,被称作文件过滤器,当中定义了一个抽象方法accept(File dir,String name),在调用list()方法时,需要实现文件过滤器,在accept()方法中做出判断,从而获得指定类型的文件。 为了让初学者更好地理解文件过滤的原理,接下来分步骤分析list(FilenameFilter filter)方法的工作原理。 - 调用list()方法传入FilenameFilter文件过滤器对象。 - 取出当前File对象所代表目录下的所有子目录和文件。 - 对于每一个子目录或文件,都会调用文件过滤器对象的accept(File dir,String name)方法,并把代表当前目录的File对象以及这个子目录或文件的名字作为参数dir和name传递给方法。 - 如果accept()方法返回true,就将当前遍历的这个子目录或文件添加到数组中,如果返回false,则不添加。 接下来通过一个案例来演示如何遍历指定目录下所有扩展名为.java的文件,如例所示。 ```java import java.io.File; import java.io.FilenameFilter; public class IOTest { public static void main(String[] args) { // 创建File 对象 File file = new File("D:/test"); // 创建过滤器对象 FilenameFilter filter = new FilenameFilter() { // 实现accept()方法 public boolean accept(File dir, String name) { File currFile = new File(dir, name); // 如果文件名以.java 结尾返回true,否则返回false if (currFile.isFile() && name.endsWith(".java")) { return true; } else { return false; } } }; if (file.exists()) { // 判断File 对象对应的目录是否存在 String[] lists = file.list(filter); // 获得过滤后的所有文件名数组 for (String name : lists) { System.out.println(name); } } } } ``` 例的main()方法中,定义了FilenameFilter文件过滤器对象filter,并且实现了accept()方法,在accept()方法中对当前正在遍历的currFile对象进行判断,只有当currFile对象代表文件,并且扩展名“.java”时,才返回true。在调用File对象的list()方法时将filter过滤器对象传入,就得到包含所有“.java”文件名字的字符串数组。 前面的两个例子演示的都是遍历目录下文件的文件名,有时候在一个目录下,除了文件,还有子目录,如果想得到所有子目录下的File类型对象,list()方法显然不能满足要求,这时需要使用File类提供的另一个方法listFiles()。listFiles()方法返回一个File对象数组,当对数组中的元素进行遍历时,如果元素中还有子目录需要遍历,则需要使用递归。接下来通过一个案例来实现遍历指定目录下的文件,如例所示。 ```java import java.io.File; public class IOTest { public static void main(String[] args) { File file = new File("D:/test"); // 创建一个代表目录的File 对象 fileDir(file); } public static void fileDir(File dir) { File[] files = dir.listFiles(); // 获得表示目录下所有文件的数组 for (File file : files) { // 遍历所有的子目录和文件 if (file.isDirectory()) { fileDir(file); // 如果是目录,递归调用FileDir() } System.out.println(file.getAbsolutePath()); // 输出文件的绝对路径 } } } ``` 运行结果: ``` D:\test\one D:\test\test.txt D:\test\two ``` 例中,定义了一个静态方法fileDir(),方法接收一个表示目录的File对象。在方法中,首先通过调用listFiles()方法把该目录下所有的子目录和文件存到一个File类型的数组files中,接着遍历数组files,对当前遍历的File对象进行判断,如果是目录就重新调用fileDir()方法进行递归,如果是文件就直接打印输出文件的路径,这样该目录下的所有文件就被成功遍历出来了。 #### 删除文件及目录 在操作文件时,经常需要删除一个目录下的某个文件或者删除整个目录,这时大家首先会想到File类的delete()方法,接下来通过一个案例来演示使用delete()方法删除文件,如例所示。 ```java import java.io.File; public class IOTest { public static void main(String[] args) { File file = new File("D:/test"); // 这是一个代表目录的File 对象 if (file.exists()) { System.out.println(file.delete()); } } } ``` 运行结果: ``` false ``` 图的运行结果中输出了false,这说明删除文件失败了。大家可能会疑惑,为什么会失败呢? 那是因为File类的delete()方法只是删除一个指定的文件,假如File对象代表目录,并且目录下包含子目录或文件,则File类的delete()方法不允许对这个目录直接删除。在这种情况下,需要通过递归的方式将整个目录以及其中的文件全部删除,接下来通过一个案例来演示,如例所示。 ```java import java.io.File; public class IOTest { public static void main(String[] args) { File file = new File("D:/test"); // 创建一个代表目录的File 对象 deleteDir(file); // 调用deleteDir 删除方法 } public static void deleteDir(File dir) { if (dir.exists()) { // 判断传入的File 对象是否存在 File[] files = dir.listFiles(); // 得到File 数组 for (File file : files) { // 遍历所有的子目录和文件 if (file.isDirectory()) { deleteDir(file); // 如果是目录,递归调用deleteDir() } else { // 如果是文件,直接删除 file.delete(); } } // 删除完一个目录里的所有文件后,就删除这个目录 dir.delete(); } } } ``` 例中,定义了一个删除目录的静态方法deleteDir(),接收一个File类型的参数。在这个方法中,调用listFiles()方法把这个目录下所有的子目录和文件保存到一个File类型的数组files中,然后遍历files,如果是目录就重新调用deleteDir()方法进行递归,如果是文件就直接调用File的delete()方法删除。当删除完一个目录下的所有文件后,再删除当前这个目录,这样便从里层到外层递归地删除了整个目录。 需要注意的是,在Java中删除目录是从虚拟机直接删除而不走回收站,文件将无法恢复,因此在进行删除操作的时候需要格外小心。 ## 字符编码 #### 常用字符集 看战争片时,经常会看到剧中出现收发电报的情况,发报员拿着密码本将文字翻译成某种码文发出,收报员使用同样的密码本将收到的码文再翻译成文字。这个密码本其实是发送方和接收方约定的一套电码表,电码表中规定了文字和电码之间的一一对应关系。 在计算机之间,同样无法直接传输一个一个的字符,而只能传输二进制数据。为了使发送的字符信息能以二进制数据的形式进行传输,同样需要使用一种“密码本”,它叫做字符码表。字符码表是一种可以方便计算机识别的特定字符集,它是将每一个字符和一个唯一的数字对应而形成的一张表。针对不同的文字,每个国家都制定了自己的码表,下面就来介绍几种最常用的字符码表,如表所示。 ![](http://47.107.171.232/easily-j/images/20190111/86838adf-7d69-4ac1-b642-54cd02c634d5.png) 表中列举了最常用的几种码表,通过选择合适的码表就能完成字符和二进制数据之间的转换,从而实现数据的传输。