1. Java 流(Stream)、文件(File)和IO
- Java.io 包几乎包含了所有操作输入、输出需要的类。所有这些流类代表了输入源和输出目标。
- Java.io 包中的流支持很多种格式,比如:基本类型、对象、本地化字符集等等。
一个流可以理解为一个数据的序列。
- 输入流表示从一个源读取数据,输出流表示向一个目标写数据。
Java 为 I/O 提供了强大的而灵活的支持,使其更广泛地应用到文件传输和网络编程中。
! 但本节讲述最基本的和流与 I/O 相关的功能。我们将通过一个个例子来学习这些功能。
1.1 读取控制台输入:
- Java 的控制台输入由
System.in
完成。
为了获得一个绑定到控制台的字符流,你可以把 System.in 包装在一个 BufferedReader 对象中来创建一个字符流。
下面是创建 BufferedReader 的基本语法:
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
BufferedReader 对象创建后,我们便可以使用 read() 方法从控制台读取一个字符,或者用 readLine() 方法读取一个字符串。
从控制台读取多字符输入:
从 BufferedReader 对象读取一个字符要使用 read() 方法,它的语法如下:
int read( ) throws IOException
- 每次调用 read() 方法,它从输入流读取一个字符并把该字符作为整数值返回。
- 当流结束的时候返回 -1。该方法抛出 IOException。
从控制台读取字符串:
- 从标准输入读取一个字符串需要使用 BufferedReader 的 readLine() 方法。
它的一般格式是:
String readLine( ) throws IOException
JDK 5 后的版本我们也可以使用 Java Scanner 类来获取控制台的输入。
1.2 控制台输出:
在此前已经介绍过,控制台的输出由 print( ) 和 println() 完成。
-
这些方法都由类 PrintStream 定义,System.out 是该类对象的一个引用。
-
PrintStream 继承了 OutputStream类,并且实现了方法 write();
这样,write() 也可以用来往控制台写操作。
PrintStream 定义 write() 的最简单格式如下所示:
void write(int byteval)
该方法将 byteval 的低八位字节写到流中。
1.3 读取文件:
- 输入流用于从源读取数据,输出流用于向目标写数据。
相关图片请查找IO流分类图;
1.3.1 FileInputStream:
- 该流用于从文件读取数据,它的对象可以用关键字 new 来创建。
- 有多种构造方法可用来创建对象。
1、可以使用字符串类型的文件名来创建一个输入流对象来读取文件:
InputStream f = new FileInputStream("C:/java/hello");
2、也可以使用一个文件对象来创建一个输入流对象来读取文件。我们首先得使用 File() 方法来创建一个文件对象
File f = new File("C:/java/hello"); InputStream out = new FileInputStream(f);
- 创建了
InputStream
对象,就可以使用下面的方法来读取流或者进行其他的流操作。
//关闭此文件输入流并释放与此流有关的所有系统资源。抛出IOException异常。
public void close() throws IOException{}
//这个方法清除与该文件的连接。确保在不再引用文件输入流时调用其 close 方法。抛出IOException异常。
protected void finalize()throws IOException {}
/**
这个方法从 InputStream 对象读取指定字节的数据。
返回为整数值。返回下一字节数据,如果已经到结尾则返回-1。
*/
public int read(int r)throws IOException{}
// 这个方法从输入流读取r.length长度的字节。返回读取的字节数。如果是文件结尾则返回-1。
public int read(byte[] r) throws IOException{}
// 返回下一次对此输入流调用的方法可以不受阻塞地从此输入流读取的字节数。返回一个整数值。
public int available() throws IOException{}
1.3.2 FileOutputStream:
- 该类用来创建一个文件并向文件中写数据。
如果该流在打开文件进行输出前,目标文件不存在,那么该流会创建该文件。
- 有两个构造方法可以用来创建 FileOutputStream 对象。
-
使用字符串类型的文件名来创建一个输出流对象:
OutputStream f = new FileOutputStream("C:/java/hello")
-
也可以使用一个文件对象来创建一个输出流来写文件。
我们首先得使用File()方法来创建一个文件对象:File f = new File("C:/java/hello"); OutputStream f = new FileOutputStream(f);
-
- 创建OutputStream 对象完成后,就可以使用下面的方法来写入流或者进行其他的流操作。
// 关闭此文件输入流并释放与此流有关的所有系统资源。抛出IOException异常。
public void close() throws IOException{}
// 这个方法清除与该文件的连接。确保在不再引用文件输入流时调用其 close 方法。抛出IOException异常。
protected void finalize()throws IOException {}
// 这个方法把指定的字节写到输出流中。
public void write(int w)throws IOException{}
// 把指定数组中w.length长度的字节写到OutputStream中。
public void write(byte[] w)
下面是一个演示 InputStream 和 OutputStream 用法的例子:
import java.io.*;
public class fileStreamTest {
public static void main(String args[]) {
try {
byte bWrite[] = { 11, 21, 3, 40, 5 };
OutputStream os = new FileOutputStream("test.txt");
for (int x = 0; x < bWrite.length; x++) {
os.write(bWrite[x]); // writes the bytes
}
os.close();
InputStream is = new FileInputStream("test.txt");
int size = is.available();
for (int i = 0; i < size; i++) {
System.out.print((char) is.read() + " ");
}
is.close();
} catch (IOException e) {
System.out.print("Exception");
}
}
}
上面的程序首先创建文件test.txt,并把给定的数字以二进制形式写进该文件,同时输出到控制台上。
以上代码由于是二进制写入,可能存在乱码,你可以使用以下代码实例来解决乱码问题:
import java.io.*;
public class fileStreamTest2 {
public static void main(String[] args) throws IOException {
File f = new File("a.txt");
FileOutputStream fop = new FileOutputStream(f);
// 构建FileOutputStream对象,文件不存在会自动新建
OutputStreamWriter writer = new OutputStreamWriter(fop, "UTF-8");
// 构建OutputStreamWriter对象,参数可以指定编码,默认为操作系统默认编码,windows上是gbk
writer.append("中文输入");
// 写入到缓冲区
writer.append("\r\n");
// 换行
writer.append("English");
// 刷新缓存冲,写入到文件,如果下面已经没有写入的内容了,直接close也会写入
writer.close();
// 关闭写入流,同时会把缓冲区内容写入文件,所以上面的注释掉
fop.close();
// 关闭输出流,释放系统资源
FileInputStream fip = new FileInputStream(f);
// 构建FileInputStream对象
InputStreamReader reader = new InputStreamReader(fip, "UTF-8");
// 构建InputStreamReader对象,编码与写入相同
StringBuffer sb = new StringBuffer();
while (reader.ready()) {
sb.append((char) reader.read());
// 转成char加到StringBuffer对象中
}
System.out.println(sb.toString());
reader.close();
// 关闭读取流
fip.close();
// 关闭输入流,释放系统资源
}
}
1.4 创建目录:
File类中有两个方法可以用来创建文件夹:
-
mkdir()
- 方法创建一个文件夹,成功则返回true,失败则返回false。
- 失败表明File对象指定的路径已经存在,或者由于整个路径还不存在,该文件夹不能被创建。
-
mkdirs()
- 方法创建一个文件夹和它的所有父文件夹。
下面的例子创建 "/tmp/user/java/bin"文件夹:
import java.io.File;
public class CreateDir {
public static void main(String args[]) {
String dirname = "/tmp/user/java/bin";// 接收路径
File d = new File(dirname);// 创建file对象
d.mkdirs();// 现在创建目录
}
}
注意:Java 在 UNIX 和 Windows 自动按约定分辨文件路径分隔符。如果你在 Windows 版本的 Java 中使用分隔符 (/) ,路径依然能够被正确解析。
1.5 读取目录:
-
一个目录其实就是一个 File 对象,它包含其他文件和文件夹。
-
如果创建一个 File 对象并且它是一个目录,那么调用 isDirectory() 方法会返回 true。
-
可以通过调用该对象上的 list() 方法,来提取它包含的文件和文件夹的列表。
下面展示的例子说明如何使用 list() 方法来检查一个文件夹中包含的内容:
import java.io.File;
public class DirList {
public static void main(String args[]) {
String dirname = "/tmp";
File f1 = new File(dirname);
if (f1.isDirectory()) {
System.out.println("目录 " + dirname);
String s[] = f1.list();
for (int i = 0; i < s.length; i++) {
File f = new File(dirname + "/" + s[i]);
if (f.isDirectory()) {
System.out.println(s[i] + " 是一个目录");
} else {
System.out.println(s[i] + " 是一个文件");
}
}
} else {
System.out.println(dirname + " 不是一个目录");
}
}
}
以上实例编译运行结果如下:
目录 /tmp
bin 是一个目录
lib 是一个目录
demo 是一个目录
test.txt 是一个文件
README 是一个文件
index.html 是一个文件
include 是一个目录
1.6 删除目录或文件
删除文件可以使用 java.io.File.delete()
方法。
以下代码会删除目录 /tmp/java/,需要注意的是当删除某一目录时,必须保证该目录下没有其他文件才能正确删除,否则将删除失败。
测试目录结构:
/tmp/java/
|-- 1.log
|-- test
import java.io.File;
public class DeleteFileDemo {
public static void main(String args[]) {
// 这里修改为自己的测试目录
File folder = new File("/tmp/java/");
deleteFolder(folder);
}
// 删除文件及目录
public static void deleteFolder(File folder) {
File[] files = folder.listFiles();// 返回folder该路径下的全部文件包括文件夹
if (files != null) {
for (File f : files) {
if (f.isDirectory())/* 确保该路径下是一个文件夹 */ {
deleteFolder(f);//重新调用自身方法
} else {
f.delete();// 删除文件
}
}
}
folder.delete();
}
}
2. ❤️Java基础专栏 - 前篇回顾
- 认识Java,Java程序的生命周期,运行Java程序
- Java数据类型阐述、基本数据类型的占用和范围、二进制的讲述
- 8种基本数据类型的分析、数据类型转换规则、转义字符的列举
- 五种运算符的说明(&&、||、>=、>、<=、<)、4种控制语句(if、for、while、dowhile)、输入和输出说明
- 方法的概念、方法的调用、方法重载、构造方法的创建
- 全方面带你了解Java里的日期与时间内容,介绍 Calendar、GregorianCalendar、Date类
- 正则表达式的使用与常用类分享