Java基础IO类
一、File类
File类是java.io包中唯一对文件本身进行操作的类。它可以进行创建、删除文件等操作。
1. 创建文件
可以使用 createNewFille() 创建一个新文件。
2. 删除文件
可以使用 delete() 删除一个文件。
演示创建和删除文件操作
import java.io.File;
import java.io.IOException;
public class FileDemo01 {
public static void createFile(String filepath) {
File f = new File(filepath);
System.out.println("[Create File] " + f.getPath().toString());
try {
f.createNewFile(); // 创建文件,根据给定的路径创建
} catch (IOException e) {
e.printStackTrace(); // 输出异常信息
}
}
public static void deleteFile(String filepath) {
File f = new File(filepath);
System.out.println("[Delete File] " + f.getPath().toString());
if(f.exists()){ // 如果文件存在则删除
f.delete(); // 创建文件,根据给定的路径创建
}
}
public static void main(String args[]) {
// 注意三种分隔符方式
String path1 = "d:\\test1.txt"; // Windows中使用反斜杠表示目录的分隔符"\"
String path2 = "d:/test2.txt"; // Linux中使用正斜杠表示目录的分隔符"/"
String path3 = "d:" + File.separator + "test3.txt"; // 最好的做法是使用File.separator,可以根据所在操作系统选取对应分隔符
createFile(path1);
createFile(path2);
createFile(path3);
deleteFile(path1);
}
};
3. 创建文件夹
可以使用 mkdir() 来创建文件夹,但是如果要创建的目录的父路径不存在,则无法创建成功。
public class FileDemo02 {
public static void main(String args[]) {
File f = new File("d:\\abc\\test"); // 实例化File类的对象
f.mkdir(); // 创建文件夹
// f.mkdirs(); // 创建文件夹,如果父路径不存在,会一并创建
}
};
4. 列出指定目录全部文件
File中给出了两种列出文件夹内容的方法:
public class FileDemo03 {
public static void testListFiles1(String path) {
File f = new File(path); // 实例化File类的对象
String str[] = f.list(); // 列出给定目录中的内容
System.out.println("[list]");
for (int i = 0; i < str.length; i++) {
System.out.println(str[i]);
}
}
public static void testListFiles2(String path) {
File f = new File(path); // 实例化File类的对象
File files[] = f.listFiles(); // 列出全部内容
System.out.println("[listFiles]");
for (int i = 0; i < files.length; i++) {
System.out.println(files[i]);
}
}
public static void main(String args[]) {
String path = "d:" + File.separator;
testListFiles1(path);
testListFiles2(path);
}
};
5. 删除目录
可以使用 delete() 来删除目录。
import java.io.File;
public class DeleteDirectory {
/**
* 删除空目录,如果目录不为空,无法删除
* @param dir
*/
private static void deleteEmptyDir(String dir) {
File f = new File(dir);
boolean success = f.delete();
if (success) {
System.out.println("Success to deleted " + dir);
} else {
System.out.println("Failed to delete " + dir);
}
}
/**
* 如果传入的是一个目录对象,遍历删除其所有子文件和子目录
* @param dir
* @return
*/
private static boolean deleteDir(File dir) {
if (dir.isDirectory()) {
File[] children = dir.listFiles();// 递归删除目录中的子目录下
for (int i = 0; i < children.length; i++) {
boolean success = deleteDir(children[i]);
if (!success) {
return false;
}
}
}
// 目录此时为空,可以删除
return dir.delete();
}
public static void main(String[] args) {
String emptyDir = "d:\\empty";
String dir2 = "d:\\test";
deleteEmptyDir(emptyDir);
boolean success = deleteDir(new File(dir2));
if (success) {
System.out.println("Success to deleted " + dir2);
} else {
System.out.println("Failed to delete " + dir2);
}
}
}
二、RandomAccessFile类
RandomAccessFile类是随机读取类,它是一个完全独立的类。
它适用于由大小已知的记录组成的文件,所以我们可以使用seek()将记录从一处转移到另一处,然后读取或者修改记录。
文件中记录的大小不一定都相同,只要能够确定哪些记录有多大以及它们在文件中的位置即可。
1. 写入数据
当用 rw 方式声明RandomAccessFile对象时,如果要写入的文件不存在,系统将自行创建。
为了保证可以进行随机读取,所有写入的名字都是8个字节,写入的数字都是固定的4个字节。
import java.io.File;
import java.io.RandomAccessFile;
public class RandomAccessFileDemo01 {
// 所有的异常直接抛出,程序中不再进行处理
public static void main(String args[]) throws Exception {
File f = new File("d:" + File.separator + "test.txt"); // 指定要操作的文件
RandomAccessFile rdf = null; // 声明RandomAccessFile类的对象
rdf = new RandomAccessFile(f, "rw");// 读写模式,如果文件不存在,会自动创建
String name = null;
int age = 0;
name = "zhangsan"; // 字符串长度为8
age = 30; // 数字的长度为4
rdf.writeBytes(name); // 将姓名写入文件之中
rdf.writeInt(age); // 将年龄写入文件之中
name = "lisi "; // 字符串长度为8
age = 31; // 数字的长度为4
rdf.writeBytes(name); // 将姓名写入文件之中
rdf.writeInt(age); // 将年龄写入文件之中
name = "wangwu "; // 字符串长度为8
age = 32; // 数字的长度为4
rdf.writeBytes(name); // 将姓名写入文件之中
rdf.writeInt(age); // 将年龄写入文件之中
rdf.close(); // 关闭
}
};
2. 读取数据
读取是直接使用 r 的模式即可,以只读的方式打开文件。
import java.io.File;
import java.io.RandomAccessFile;
public class RandomAccessFileDemo02{
// 所有的异常直接抛出,程序中不再进行处理
public static void main(String args[]) throws Exception{
File f = new File("d:" + File.separator + "test.txt") ; // 指定要操作的文件
RandomAccessFile rdf = null ; // 声明RandomAccessFile类的对象
rdf = new RandomAccessFile(f,"r") ;// 以只读的方式打开文件
String name = null ;
int age = 0 ;
byte b[] = new byte[8] ; // 开辟byte数组
// 读取第二个人的信息,意味着要空出第一个人的信息
rdf.skipBytes(12) ; // 跳过第一个人的信息
for(int i=0;i<b.length;i++){
b[i] = rdf.readByte() ; // 读取一个字节
}
name = new String(b) ; // 将读取出来的byte数组变为字符串
age = rdf.readInt() ; // 读取数字
System.out.println("第二个人的信息 --> 姓名:" + name + ";年龄:" + age) ;
// 读取第一个人的信息
rdf.seek(0) ; // 指针回到文件的开头
for(int i=0;i<b.length;i++){
b[i] = rdf.readByte() ; // 读取一个字节
}
name = new String(b) ; // 将读取出来的byte数组变为字符串
age = rdf.readInt() ; // 读取数字
System.out.println("第一个人的信息 --> 姓名:" + name + ";年龄:" + age) ;
rdf.skipBytes(12) ; // 空出第二个人的信息
for(int i=0;i<b.length;i++){
b[i] = rdf.readByte() ; // 读取一个字节
}
name = new String(b) ; // 将读取出来的byte数组变为字符串
age = rdf.readInt() ; // 读取数字
System.out.println("第三个人的信息 --> 姓名:" + name + ";年龄:" + age) ;
rdf.close() ; // 关闭
}
};
三、系统标准IO
System表示系统类,它有3个与 Java IO 有关的常量。
- System.out——系统标准输出
- System.in——系统标准输入
- System.err——错误信息输出
1. System.out
System.out 是 PrintStream 的对象,在 PrintStream 中定义了一些了的 print() 和 println() 方法。
所以,常见的 System.out.print() 或 System.out.println() 语句调用的实际上是 PrintStream 类的方法。
import java.io.OutputStream;
import java.io.IOException;
public class SystemDemo01 {
public static void main(String args[]) {
OutputStream out = System.out; // 此时的输出流是向屏幕上输出
try {
out.write("hello world!!!".getBytes()); // 向屏幕上输出
} catch (IOException e) {
e.printStackTrace(); // 打印异常
}
try {
out.close(); // 关闭输出流
} catch (IOException e) {
e.printStackTrace();
}
}
};
2. System.in
System.in 是 InputStream 类型的对象,可以利用它完成从键盘读取数据的功能。
import java.io.InputStream;
public class SystemInDemo {
public static void main(String args[]) throws Exception {
InputStream input = System.in;
byte b[] = new byte[5]; // 开辟空间,接收数据
System.out.print("请输入内容:");
int len = input.read(b); // 接收数据
System.out.println("输入的内容为:" + new String(b, 0, len));
input.close(); // 关闭输入流
}
};
运行结果
请输入内容:Good Bye
输入的内容为:Good
上述结果中,输入内容并没有被完整打印出来。
这是因为代码中限制了存储数据的 byte 数组的大小。
如果不知道要输入的数据的实际长度,如何处理呢?
不指定大小,从键盘读取数据
import java.io.InputStream;
public class SystemInDemo2 {
public static void main(String args[]) throws Exception { // 所有异常抛出
InputStream input = System.in; // 从键盘接收数据
StringBuffer buf = new StringBuffer(); // 使用StringBuffer接收数据
System.out.print("请输入内容:");
int temp = 0;
while ((temp = input.read()) != -1) { // 接收内容
char c = (char) temp;
if (c == '\n') { // 退出循环,输入回车表示输入完成
break;
}
buf.append(c); // 保存内容
}
System.out.println("输入的内容为:" + buf);
input.close(); // 关闭输入流
}
};
如果是中文会出现乱码。
这是因为汉字一个字符占两个字节,而数据却一个一个字节的方式读进来的,所以造成了乱码。
最好的输入方式是将全部输入数据暂时存放在一块内存中,然后一次性从内存中读取出数据。
这样,既不会造成乱码,也不受长度限制。要完成这样的操作可以使用 BufferedReader 类去完成。
3. System.err
System.err 表示的是错误信息输出,如果程序出现错误,则可以直接使用 System.err 进行输出。
public class SystemErrDemo {
public static void main(String args[]) {
String str = null;
try {
System.out.println(Integer.parseInt(str)); // 转型
} catch (Exception e) {
System.err.println(e);
}
}
};
4. IO重定向
System 类可以改变 System.in 的输入流来源以及 System.out 和 System.err 两个输出流的输出位置。
import java.io.*;
public class RedirectDemo {
public static void main(String[] args) throws IOException {
PrintStream console = System.out;
BufferedInputStream in = new BufferedInputStream(new FileInputStream("d:\\in.txt")); // 绑定输入文件
PrintStream out = new PrintStream(new BufferedOutputStream(new FileOutputStream("d:\\out.txt"))); // 绑定输出文件
// 设置重定向
System.setIn(in);
System.setOut(out);
System.setErr(out);
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
String s;
while ((s = br.readLine()) != null) {
System.out.println(s);
out.close();
System.setOut(console);
}
}
}
需要注意的是,I/O重定向操纵的是字节流,而不是字符流。
四、Scanner
Scanner 可以完成输入数据操作,并对数据进行验证。
示例:
import java.io.*; public class ScannerDemo { public static void main(String args[]) { Scanner scan = new Scanner(System.in); // 从键盘接收数据 int i = 0; float f = 0.0f; System.out.print("输入整数:"); if (scan.hasNextInt()) { // 判断输入的是否是整数 i = scan.nextInt(); // 接收整数 System.out.println("整数数据:" + i); } else { System.out.println("输入的不是整数!"); } System.out.print("输入小数:"); if (scan.hasNextFloat()) { // 判断输入的是否是小数 f = scan.nextFloat(); // 接收小数 System.out.println("小数数据:" + f); } else { System.out.println("输入的不是小数!"); } Date date = null; String str = null; System.out.print("输入日期(yyyy-MM-dd):"); if (scan.hasNext("^\\d{4}-\\d{2}-\\d{2}$")) { // 判断 str = scan.next("^\\d{4}-\\d{2}-\\d{2}$"); // 接收 try { date = new SimpleDateFormat("yyyy-MM-dd").parse(str); } catch (Exception e) {} } else { System.out.println("输入的日期格式错误!"); } System.out.println(date); } }
参考
https://www.cnblogs.com/jingmoxukong/p/4513059.html
https://www.cnblogs.com/jingmoxukong/p/4582641.html
https://dunwu.github.io/javacore/io/java-io-base.html#scanner