【JavaEE】文件操作: File 类的用法和 InputStream, OutputStream 的用法

2024-01-18 20:36

本文主要是介绍【JavaEE】文件操作: File 类的用法和 InputStream, OutputStream 的用法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

1. File 概述

1.1 File的属性

1.2 File的构造方法

1.3 File的方法

2.读文件

2.1 InputStream 概述

2.2 FileInputStream 概述

2.3 正确打开和关闭文件的方式

2.4 不同方式读取文件代码示例

2.4 另一种方法:利用 Scanner 进行字符读取

3.写文件

3.1 OutputStream 概述

3.2 利用 OutputStreamWriter 进行字符写入

4. 读文件和写文件的其他方式: Reader 和 Writer 

4.1 Reader 读取文件

4.2 Writer 写文件


1. File 概述

Java 中通过 java.io.File 类来对⼀个文件(包括目录)进行抽象的描述。注意,有 File 对象, 并不代表真实存在该文件

我们先来看看 File 类中的常见属性、构造方法和方法:

1.1 File的属性

修饰符及类型属性说明
static StringpathSeparator依赖于系统的路径分隔符,String 类型的表示
static charpathSeparator依赖于系统的路径分隔符,char 类 型的表示

1.2 File的构造方法

签名说明
File(File parent, String child)根据父目录 + 孩子文件路径,创建⼀个新的 File 实例
File(String pathname)根据文件路径创建⼀个新的 File 实例,路径可以是绝对路径或者相对路径
File(String parent, String child)根据父目录 + 孩子文件路径,创建⼀个新的 File 实 例,父目录用路径表示

其中 File(String pathname) 使用的最多,下面是使用实例:

File f1 = new File("C:/Users/1/test.txt");
File f2 = new File("./test.txt");

1.3 File的方法

修饰符及返回值类型方法签名说明
StringgetParent()返回 File 对象的父目录文件路径
StringgetName()返回 FIle 对象的纯文件名称
StringgetPath()返回 File 对象的文件路径
StringgetAbsolutePath()返回 File 对象的绝对路径
StringgetCanonicalPath()返回 File 对象的修饰过的绝对路径
booleanexists()判断 File 对象描述的文件是否真实存在
booleanisDirectory()判断 File 对象代表的文件是否是⼀个目录
booleanisFile()判断 File 对象代表的文件是否是⼀个普通文件
booleancreateNewFile()根据 File 对象,⾃动创建⼀个空文 件。成功创建后返回 true
booleandelete()根据 File 对象,删除该文件。成功 删除后返回 true
voiddeleteOnExit()根据 File 对象,标注文件将被删 除,删除动作会到 JVM 运行结束时 才会进行
String[]list()返回 File 对象代表的目录下的所有文件名
File[]listFiles()返回 File 对象代表的目录下的所有文件,以 File 对象表示
booleanmkdir()创建 File 对象代表的目录
booleanmkdirs()创建 File 对象代表的目录,如果必 要,会创建中间目录
booleanrenameTo(File dest)进行文件改名,也可以视为我们平时的剪切、粘贴操作
booleancanRead()判断用户是否对文件有可读权限
booleancanWrite()判断用户是否对文件有可写权限

使用示例

getParent(),getName(),getPath(),getAbsolutePath(),getCanonicalPath()

public static void main(String[] args) throws IOException {File f = new File("C:/Users/1/test.txt");//File f = new File("./test.txt");System.out.println(f.getParent());System.out.println(f.getName());System.out.println(f.getPath());System.out.println(f.getAbsolutePath());System.out.println(f.getCanonicalPath());}

exists(),isDirectory(),isFile(),createNewFile()

createNewFile(): 根据 File 对象,自动创建⼀个空文件。成功创建后返回 true

public static void main(String[] args) throws IOException {File file = new File("./test.txt");System.out.println(file.exists());System.out.println(file.isFile());System.out.println(file.isDirectory());boolean ret = file.createNewFile();System.out.println("ret = " + ret);System.out.println(file.exists());System.out.println(file.isFile());System.out.println(file.isDirectory());}

 

delete(),deleteOnExit()

delete():根据 File 对象,删除该文件。成功删除后返回 true

public static void main(String[] args) throws InterruptedException {File file = new File("./test.txt");boolean ret = file.delete();System.out.println("ret = " + ret);}

deleteOnExit():根据 File 对象,标注文件将被删除,删除动作会到 JVM 运⾏结束时才会进行

public static void main(String[] args) throws InterruptedException {File file = new File("./test.txt");      //等程序结束后再删除文件file.deleteOnExit();Thread.sleep(3000);}

list(),listFiles()

list():返回 File 对象代表的目录下的所有文件名

public static void main(String[] args) {File file = new File(".");String[] files = file.list();System.out.println(Arrays.toString(files));}

listFiles():返回 File 对象代表的目录下的所有件,以 File 对象表⽰

public static void main(String[] args) {File file = new File(".");File[] files = file.listFiles();System.out.println(Arrays.toString(files));}

mkdir():创建 File 对象代表的目录

public static void main(String[] args) {File f = new File("./aaa");//需要在构造方法中把路径创建好,再通过mkdir来创建boolean ret = f.mkdir();//boolean ret = f.mkdirs(); //创建多级目录System.out.println(ret);}

mkdirs():创建 File 对象代表的目录,如果必要,会创建中间目

public static void main(String[] args) {File f = new File("./aaa/bbb/ccc");//需要在构造方法中把路径创建好,再通过mkdir来创建boolean ret = f.mkdirs(); //创建多级目录System.out.println(ret);}

renameTo(File dest):进行文件改名,也可以视为我们平时的剪切、粘贴操作

修改文件名:

public static void main(String[] args) {File src = new File("./test.txt");File dest = new File("./test2.txt");boolean ret = src.renameTo(dest);//重命名和移动目录System.out.println(ret);}

修改路径:

public static void main(String[] args) {File src = new File("./test2.txt");File dest = new File("./aaa/test2.txt");boolean ret = src.renameTo(dest);//重命名和移动目录System.out.println(ret);}

2.读文件

2.1 InputStream 概述

方法

修饰符及返回值类型方法签名说明
intread()读取⼀个字节的数据,返回 -1 代表已经完全读完了
intread(byte[] b)最多读取 b.length 字节的数据到 b 中,返回实际读到的数量;-1 代表已经读完了
intread(byte[] b, int off, int len)最多读取 len - off 字节的数据到 b 中,从 off 开始,返回实际读到的数量;-1 代表已经读完了
voidclose()关闭字节流

byte[] b 是一个输出型参数, 会将文件的内容读取到b数组中,单位是字节. 

说明

InputStream 只是⼀个抽象类,要使⽤还需要具体的实现类。关于 InputStream 的实现类有很多,基本可以认为不同的输入设备都可以对应⼀个 InputStream 类,我们现在只关心从文件中读取,所以使用 FileInputStream

2.2 FileInputStream 概述

构造方法

签名说明
FileInputStream(File file)利⽤ File 构造文件输⼊流
FileInputStream(String name)利⽤文件路径构造文件输⼊流

2.3 正确打开和关闭文件的方式

不建议使用下面这种方式来关闭文件:

因为在文件操作的执行过程中, 程序可能会在中途退出, 如遇到 return, 或者抛出异常未正确处理等, 会导致文件无法正常关闭, 造成文件资源泄露.

文件资源泄露:在java PCB中有文件描述符表, 记录了当前进程都打开了哪些文件, 每次打开一个文件, 都是需要在文件描述符表中占据一个位置, 如果不关闭还一直打开其他文件, 就会导致文件描述符表被耗尽(文件描述符表长度有上限). 当文件描述符表被耗尽后, 后续再打开文件就会打开失败, 就会一起后续一系列的逻辑出现问题.

下面是建议使用的文件关闭方法:

1.使用finally

在javaSE中提到过finally的一个知识点:

        try {// 可能会发生异常的代码} catch (Exception e) {// 异常处理} finally {// 无论是否发生异常,都会执行的代码}

2. 使用java中提供的"try-with-resources"语句

Java中提供了"try-with-resources"语句,用于自动关闭实现了"AutoCloseable"接口的资源,无需显式调用close()方法来关闭资源。这个语句可以在代码块结束后自动关闭资源,无论代码块是正常执行完毕还是发生了异常。

 

2.4 不同方式读取文件代码示例

1.每次读取一个字节, 以16进制输出:

public static void main(String[] args) throws IOException{try(InputStream inputStream = new FileInputStream("./test.txt")) {//进行文件操作while(true) {int n = inputStream.read();if(n == -1) {//文件读取完毕break;}//打印这个字节的数据System.out.printf("%x ",n);}}}

 

2.一次读取若干个字节,以String字符串的形式输出:

public static void main(String[] args) throws IOException{try(InputStream inputStream = new FileInputStream("./test.txt")) {//进行文件操作while(true) {byte[] buffer = new byte[1024];int n = inputStream.read(buffer);if(n == -1) {//文件读取完毕break;}//打印这个字节的数据String s = new String(buffer,0,n);System.out.print(s);}}}

2.4 另一种方法:利用 Scanner 进行字符读取

上述例子中,我们看到了对字符类型直接使用 InputStream 进行读取是非常麻烦且困难的,所以,我们使用一种我们之前比较熟悉的类来完成该工作,就是 Scanner 类。

public static void main(String[] args) {try(InputStream inputStream = new FileInputStream(",/test.txt")) {Scanner scanner = new Scanner(inputStream);while(scanner.hasNext()) {String s = scanner.next();System.out.println(s);}} catch (IOException e) {throw new RuntimeException(e);}}

3.写文件

3.1 OutputStream 概述

方法

返回值类型

⽅法签名说明
voidwrite(int b)写入一个字节的数据
voidwrite(byte[] b)将 b 这个字符数组中的数据全部写入 os 中
intwrite(byte[] b, int off, int len)将 b 这个字符数组中从 off 开始的数据写入 os 中,⼀共写 len 个
voidclose()关闭字节流
voidflush()见下面

flush()重要:我们知道 I/O 的速度是很慢的,所以,大多的 OutputStream 为了减少设备操作的次数,在写数据的时候都会将数据先暂时写入内存的⼀个指定区域里,直到该区域满了或者其他指定条件时才真正将数据写入设备中,这个区域⼀般称为缓冲区。但造成⼀个结果,就是我们写的数据,很可能会遗留⼀部分在缓冲区中。需要在最后或者合适的位置,调用 flush(刷新)操作,将数据刷到设备中.

说明

OutputStream 同样只是⼀个抽象类,要使⽤还需要具体的实现类。我们现在还是只关心写入文件 中,所以使⽤ FileOutputStream

3.2 利用 OutputStreamWriter 进行字符写入

public static void main(String[] args) {//使用OutPutStream来写文件try(OutputStream outputStream = new FileOutputStream("./test.txt")) {byte[] buffer = new byte[]{97,98,99,100,101,102};outputStream.write(buffer);// 不要忘记 flushoutputStream.flush();} catch (IOException e) {throw new RuntimeException(e);} ;}

 文件中得到的结果:

注意: 

如果要以追加的形式写文件,需要在FileOutputStream的构造方法中加true参数:

写入英文字符串

public static void main(String[] args) {//使用OutPutStream来写文件try(OutputStream outputStream = new FileOutputStream("./test.txt")) {String s = "Nothing";byte[] buffer = s.getBytes();outputStream.write(buffer);// 不要忘记 flushoutputStream.flush();} catch (IOException e) {throw new RuntimeException(e);} }

写入中文字符串

public static void main(String[] args) {//使用OutPutStream来写文件try(OutputStream outputStream = new FileOutputStream("./test.txt")) {String s = "你好中国";byte[] buffer = s.getBytes("utf-8");outputStream.write(buffer);// 不要忘记 flushoutputStream.flush();} catch (IOException e) {throw new RuntimeException(e);}}

4. 读文件和写文件的其他方式: Reader 和 Writer 

Reader 和 Writer 的使用方式和 InputStream,OutputStream 类似, 具体使用方式可以参考这两个

4.1 Reader 读取文件

方法

使用示例:  

public static void main(String[] args) {//使用字符流读取文件内容try(Reader reader = new FileReader("./test.txt")) {while(true) {char[] buffer = new char[1024];int n = reader.read(buffer);if(n == -1) {break;}String s = new String(buffer,0,n);System.out.println(s);}} catch (IOException e) {throw new RuntimeException(e);}}



4.2 Writer 写文件

方法 

使用示例: 

public static void main(String[] args) {try(Writer writer = new FileWriter("./test.txt")) {String s = "你好";writer.write(s);} catch (IOException e) {throw new RuntimeException(e);}}

 如果要以追加的形式写文件, 只需要在在构造方法中加一个true即可:

Writer writer = new FileWriter("./test.txt",true)

这篇关于【JavaEE】文件操作: File 类的用法和 InputStream, OutputStream 的用法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/620179

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟 开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚 第一站:海量资源,应有尽有 走进“智听

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定