Java文件管理
ren2003u 人气:1一.基本文件操作
获取及判断文件属性
代码示例如下:
import java.io.IOException; public class test { public static void main(String[] args) throws IOException { File file = new File("./test.txt"); System.out.println(file.getParent()); System.out.println(file.getName()); System.out.println(file.getPath()); System.out.println(file.getAbsolutePath()); System.out.println(file.getCanonicalPath()); //创建文件前 System.out.println(file.exists()); System.out.println(file.isDirectory()); System.out.println(file.isFile()); System.out.println("---------------"); //创建文件后 file.createNewFile(); System.out.println(file.exists()); System.out.println(file.isDirectory()); System.out.println(file.isFile()); //删除文件 System.out.println("--------------"); file.delete(); System.out.println(file.exists()); } }
getParent():获取上级目录,若无指定则返回null
getName():获取当前文件名
getPath():获取以工作目录为基准的相对路径
getAbsolutePath():获取以盘头为起始点的绝对路径
getCanonicalPath():同上,但是会进行一些精简
exists():判断文件是否存在
isDirectory():判断文件是否是目录
isFile():判断文件是否是文件
delete():删除文件
上述代码的运行结果如下:
创建及修改文件
import java.io.File; import java.io.IOException; public class test { public static void main(String[] args) throws IOException { File file = new File("./test.txt"); file.createNewFile(); //退出时删除 file.deleteOnExit(); System.out.println(file.exists()); } }
deleteOnExit():退出时删除
createNewFile():创建对应文件
public class test { public static void main(String[] args) throws IOException { File file = new File("./test"); //创建单级目录 file.mkdir(); //创建多级目录 File file1 = new File("test1/a/11"); file1.mkdirs(); } }
上述代码运行结果如下:
mkdir():根据路径创建单级目录
mkdirs():根据路径创建多级目录
public class test { public static void main(String[] args) throws IOException { File file1 = new File("test"); File file2 = new File("test1"); //重命名 file1.renameTo(file2); } }
renameTo():将一个文件的名字赋值给另一个文件
二.文件读写
读文件
字节流
我们事先准备一个test.txt文件,里面包含内容"Hello".
public class test { public static void main(String[] args) throws IOException { File file = new File("text1.txt"); file.createNewFile(); //打开文件 InputStream inputStream = new FileInputStream("text1.txt"); //读文件 while(true){ int b = inputStream.read(); if(b == -1){ break; } System.out.println(b); } //关闭文件 inputStream.close(); } }
实际上,read()有很多个版本,具体如下:
上述代码中使用的是第一个版本,一次读取一个字节,并将这个字节的内容作为返回值返回,如果读取到文件结束符(EOF),则返回-1.
第二个版本要求一个"输出型参数"---一个字节数组.read()会从文件里把内容全部读入该字节数组中,若溢出则截断.
第三个版本与第二个版本相似,只不过允许通过参数控制数据的传入起点和终点(也即允许传入的数据长度).
上述代码的运行结果如下:
可见读取的是"Hello"每个字符对应的ASCII码值.
这样可读性无疑是很低的,我们可以将其转为字符串的形式输出,代码示例如下:
public class test { public static void main(String[] args) throws IOException { File file = new File("text1.txt"); file.createNewFile(); //打开文件 InputStream inputStream = new FileInputStream("text1.txt"); //读文件 byte[] b = new byte[1024]; int len = inputStream.read(b); String s = new String(b,0,len,"utf8"); System.out.println(s); //关闭文件 inputStream.close(); } }
我们可以通过读出来的字节数组,反向构造出其原本的字符串,这样就能得到"Hello"了,但这样无疑很麻烦.因此,对于这种非二进制的数组,我们可以使用字符流.
字符流
字符流对应着Reader和FileReader两个关键字.
public class test { public static void main(String[] args) throws IOException { File file = new File("text1.txt"); file.createNewFile(); //打开文件 Reader reader = new FileReader("text1.txt"); char[] buffer = new char[1024]; int len = reader.read(buffer); for (int i = 0; i < len; i++) { System.out.print(buffer[i]); } reader.close(); } }
这样输出出来的直接就是"Hello",而不是ASCII码值.
其他要点
1.InputStream和Reader两个都是抽象类,没办法直接实例化.因此我们需要借助他们的子类FileInputStream和FileReader来实例化.
2.我们创建的文件都在硬盘上,直接操作的话比较困难.因此我们尝试在内存上创建一个媒介,间接的操作硬盘上的文件.我们将InputStream和Reader这种媒介成为句柄(Handler).
3.上述代码的写法实际上是不严谨的,因为一旦程序在运行途中抛出了异常,代码末尾的close()就无法执行.因此我们应该把close()放在finally下.保证在文件描述符表上的资源得以释放.
4.关于文件描述符表:文件描述符表可以简单理解成PCB中的一个数组/顺序表.数组中的每个元素都对应着当前进程打开的文件.这个数组的下标,就称为"文件描述符".每当我们打开一个文件时,就会在文件描述符表中占据一个位置;每次关闭文件,都会释放一个位置.然而文件描述表的长度是存在上限的,如果在进程中一直打开文件而不释放,这就会导致进程在后续打开文件的时候抛出异常.
5.实际上,我们有一套更实用的方法从文件中读取内容,代码示例如下:
public class test { public static void main(String[] args) throws IOException { File file = new File("text1.txt"); file.createNewFile(); InputStream inputStream = new FileInputStream("text1.txt"); Scanner scanner = new Scanner(inputStream); String s = scanner.next(); System.out.println(s); } }
我们可以将InputStream作为Scanner构造函数的参数,这样我们就可以使用Scanner灵活读取文件内部的内容.
写文件
字节流
对于字节流的输入方式,我们有OutputStream和FileOutputStream这一套组合.
public class test { public static void main(String[] args) throws IOException { File file = new File("text1.txt"); file.createNewFile(); try(OutputStream outputStream = new FileOutputStream("text1.txt")){ outputStream.write('h'); outputStream.write('e'); outputStream.write('l'); outputStream.write('l'); outputStream.write('o'); }catch(IOException e){ e.printStackTrace(); } } }
字符流
对于字符流的输入方式,我们有Writer和FileWriter这一套组合.
public class test { public static void main(String[] args) throws IOException { File file = new File("text1.txt"); file.createNewFile(); try(Writer writer = new FileWriter("text1.txt")){ writer.write("hello"); }catch(IOException e){ e.printStackTrace(); } } }
字节流封装
与读取文件内容部分一样,我们可以将字节流封装,其可以灵活的向文件内写内容.
public class test { public static void main(String[] args) throws IOException { File file = new File("text.txt"); file.createNewFile(); try(OutputStream outputStream = new FileOutputStream("text.txt")){ PrintWriter printWriter = new PrintWriter(outputStream); printWriter.println("hello"); printWriter.flush(); }catch (IOException e){ e.printStackTrace(); } } }
此处封装过后,就可以如常规输出一样,向文件里输入内容.代码中flush()的作用在于清空输入缓冲区的内容,使println()输出的内容能成功到文件中.
其他要点
上述代码中我们把OutputStream等输入输出流放在try()中,其目的是在代码结束后可以自动调用close()方法释放文件描述符表,防止忘记.这要求这个类要实现Closeable接口.
加载全部内容