【JavaEE】——文件IO

种地  金牌会员 | 2024-10-6 14:51:09 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 878|帖子 878|积分 2634


  
阿华代码,不是逆风,就是我疯

  
你们的点赞收藏是我前进最大的动力!!

  
希望本文内容能够帮助到你!!

  目录
一:认识文件
1:文件的概念
2:文件的结构
3:文件路径
(1)绝对路径
(2)相对路径
(3)举例
二:文件的分类
1:文本文件
2:二进制文件
3:如何判断文件的种类
三:Feil类对文件利用
1:Feil类
2:File类的构造方法
3:获取属性方法
代码示例一
代码示例二
 4:判断方法
 5:创建删除方法
6:创建移动方法
7:创建File对象代表的目录
8:文件改名利用
四:文件流
五:字节流
1:InputStream
(0)打开文件
(1)构造方法
(2)类/非常解析
(3)文件资源泄漏
(4):read方法
①引入
②三种参数代码示例
2:OutputStream
3:Scanner联动
​编辑
六:字符流
1:Reader类
2:Writer类 


一:认识文件

1:文件的概念

对于计算机来说,“文件”是一个非常广义的概念,可以指普通的文件,文件夹,一些硬件设备也被抽象成了文件(比方:键盘,耳机,表现器,网卡等)
对于机器硬盘来说它更加擅长次序读写,不擅长随机读写
好比有一个很大的文件,我们需要把这个文件整体复制一份就是次序读写
某个目录中有很多的小文件,(数量比力多)需要把整个目录都拷贝一份,就是随机读写。
2:文件的结构

文件系统的结构是按照“树形结构”来构造文件的,是N叉树,一个普通文件(非目录)就是一个叶子节点,对一个目录文件来说,里面又可以有很多的子节点
3:文件路径


形如上述的图片,从树根节点,一级一级的往下走,直到直到目标文件,这个过程中经过的所有文件目录的名字串起来,使用正斜杠——“/”,或者反斜杠——“\”分割,就构成了“路径”
注:此过程并非是遍历树,可以想象成类似于Hash这样的结构
(1)绝对路径

从树根节点 出发,一层一层到达目标文件
(2)相对路径

先指定一个“当前目录”/“基准目录”,从当前目录出发,找到目标文件
(3)举例


二:文件的分类

1:文本文件

文本文件是按照“字符串”的形式来理解文本内容的(文本文件里面的二进制内容,也都表示为字符串,可以理解为——二进制内容都是合法的字符(像字符编码,英文字母用ASCII,汉字用utf8/gbk))
举例:.java   .c    .cpp   文本文档
2:二进制文件

二进制文件没有上述的限制,储存任何数据都可以。
举例:图片,音频,可实行程序,动态库,.class文件,富文本文件
富文本文件包括不限于:word  docx excle  xlsx  power point  pptx
3:如何判断文件的种类

使用记事本打开文本文件,如果看到的是乱码,就是二进制文件
三:Feil类对文件利用

引入:在Java中,对于文件利用的Api这里有两类
针对文件系统的利用:创建,删除,重定名,
针对文件内容的利用:读文件,写文件
1:Feil类


2:File类的构造方法


3:获取属性方法


代码示例一

构造File用绝对路径
  1. import java.io.File;
  2. import java.io.IOException;
  3. public class IO1 {
  4.     public static void main(String[] args) throws IOException {
  5.         File f = new File("C:/User/1/test.txt");//绝对路径
  6.         //File f = new File("./test.txt");
  7.         System.out.println(f.exists());//判断文件目录是否存在
  8.         System.out.println(f.getParent());
  9.         System.out.println(f.getName());//返回file对象的名称
  10.         System.out.println(f.getPath());//返回file对象的文件路径
  11.         System.out.println(f.getAbsolutePath());//返回绝对路径
  12.         System.out.println(f.getCanonicalPath());//返回对象修饰过后的路径canonical规范的
  13.     }
  14. }
复制代码


代码示例二

构造对象用相对路径
.getPath()得到的就是构造方法中的参数
Canonical(读音:克挠你扣)规范的——可以理解为化简路径
  1. import java.io.File;
  2. import java.io.IOException;
  3. /**
  4. * Created with IntelliJ IDEA.
  5. * Description:
  6. * User: Hua YY
  7. * Date: 2024-10-01
  8. * Time: 19:28
  9. */
  10. public class IO1 {
  11.     public static void main(String[] args) throws IOException {
  12.         //File f = new File("C:/User/1/test.txt");//绝对路径
  13.         File f = new File("./test.txt");
  14.         System.out.println(f.exists());//判断文件目录是否存在
  15.         System.out.println(f.getParent());//返回父目录文件路径
  16.         System.out.println(f.getName());//返回file对象的名称
  17.         System.out.println(f.getPath());//返回file对象的文件路径
  18.         System.out.println(f.getAbsolutePath());//返回绝对路径
  19.         System.out.println(f.getCanonicalPath());//返回对象修饰过后的路径canonical规范的
  20.     }
  21. }
复制代码

 4:判断方法



我们可以使用file.createNewFile(如下代码),也可以直接手动在当前目录创建一个File文件——名字叫test.txt
  1. import java.io.File;
  2. import java.io.IOException;
  3. public class IODemon2 {
  4.     public static void main(String[] args) throws IOException {
  5.         File file = new File("./test.txt");
  6.         System.out.println(file.exists());//文件存在吗
  7.         System.out.println(file.isFile());//是普通文件嘛
  8.         System.out.println(file.isDirectory());//是目录吗
  9.         boolean ret = file.createNewFile();//创建文件;返回的结果就是创建成功和创建失败
  10.         System.out.println("ret:" + ret);
  11.     }
  12. }
复制代码

 5:创建删除方法

file.delete删除文件,返回值范例为boolean范例
file.deleteNewFile退出时删除文件,返回值为void
注:删除目录只能一级一级的删除
  1. import com.sun.xml.internal.ws.api.model.wsdl.WSDLOutput;
  2. import java.io.File;
  3. import java.io.IOException;
  4. /**
  5. * Created with IntelliJ IDEA.
  6. * Description:
  7. * User: Hua YY
  8. * Date: 2024-10-01
  9. * Time: 20:17
  10. */
  11. public class IODemon3 {
  12.     public static void main(String[] args) throws IOException, InterruptedException {
  13.         File file = new File("./test.txt");
  14.         //boolean ret = file.delete();//删除文件
  15.         //System.out.println(ret);
  16.         boolean ret = file.createNewFile();//创建文件
  17.         file.deleteOnExit();//返回值为void类型,退出的时候删除文件,测试的时候注意看左边的目录栏
  18.         Thread.sleep(5000);
  19.     }
  20. }
复制代码
6:创建移动方法

  1. import java.io.File;
  2. import java.util.Arrays;
  3. /**
  4. * Created with IntelliJ IDEA.
  5. * Description:
  6. * User: Hua YY
  7. * Date: 2024-10-01
  8. * Time: 20:44
  9. */
  10. public class IODemon4 {
  11.     public static void main(String[] args) {
  12.         File file = new File(".");
  13.         //返回的是对象代表的目录下所有的文件
  14.         //返回的是一个String[]类型的数组
  15.         String[] files = file.list();
  16.         System.out.println(Arrays.toString(files));
  17.         //返回的file对象代表的目录下所有的文件,不过是以file对象表示
  18.         File[] files2 = file.listFiles();
  19.         System.out.println(Arrays.toString(files2));
  20.     }
  21. }
复制代码

7:创建File对象代表的目录

  1. import java.io.File;
  2. import java.lang.reflect.Field;
  3. /**
  4. * Created with IntelliJ IDEA.
  5. * Description:
  6. * User: Hua YY
  7. * Date: 2024-10-01
  8. * Time: 20:58
  9. */
  10. public class IODemon5 {
  11.     public static void main(String[] args) {
  12.        /* File file = new File("./aaa");//创建file为对象的目录
  13.         boolean ret = file.mkdir();
  14.         System.out.println(ret);*/
  15.        /* File file = new File("./bbb");
  16.         boolean ret2 = file.mkdirs();//mkdirs(可以用于创建单个目录,也可以用于创建多级目录)
  17.         System.out.println(ret2);*/
  18.          /*File file = new File("./aaa/bbb/ccc");
  19.         boolean ret2 = file.mkdirs();//mkdirs(可以用于创建单个目录,也可以用于创建多级目录)
  20.         System.out.println(ret2);*/
  21.         File file = new File("./bbb");
  22.         boolean ret = file.delete();//删除只能一级目录一级目录的删
  23.         System.out.println(ret);
  24.     }
  25. }
复制代码
8:文件改名利用

留意细节:file1文件改名为file2对象
  1. import java.io.File;
  2. public class IODemon6 {
  3.     public static void main(String[] args) {
  4.         File file1 = new File("./aaa/bbb/ccc/bbb");
  5.         File file2 = new File("./bbb");
  6.         boolean ret = file2.renameTo(file1);//把bbb这个文件移入"./aaa/bbb/ccc"的/ccc下
  7.         System.out.println(ret);
  8.     }
  9. }
复制代码

四:文件流

什么是文件流呢?
举个例子:要写100字节的内容,我们可以分10次写每次写10字节的内容,也可以分5次来写,每次写20字节的内容;当然也可以一次性写完
“流”是利用系统提供的概念,Java标准库中对于“进行了”封装,提供了一组类来负责这些工作
我们把这些类主要分为两种:字节流和字符流
五:字节流

代表类:InputStream——输入流,OutPutStream——输出流
1:InputStream

(0)打开文件

按照读文件的方式打开一个文件
  1. import java.io.FileInputStream;
  2. import java.io.FileNotFoundException;
  3. import java.io.IOException;
  4. import java.io.InputStream;
  5. /**
  6. * Created with IntelliJ IDEA.
  7. * Description:
  8. * User: Hua YY
  9. * Date: 2024-10-02
  10. * Time: 11:46
  11. */
  12. public class IODemon7 {
  13.     public static void main(String[] args) throws IOException {
  14.         /*
  15.         //打开文件
  16.         InputStream inputStream = new FileInputStream("./test.txt");
  17.         //在中间一定会涉及一些对文件的操作之类的,如果遇到bug或者像return之类的不可控的操作,close操作可能就会执行不到了
  18.         //打开了文件最后还要关掉,打开文件操作会抛出FileNotFoundException异常
  19.         inputStream.close();
  20.         //关闭文件会抛出IOException异常,前后两者为父类子类的关系
  21.         */
  22.         /*
  23.         //第二种写法
  24.         InputStream inputStream = null;
  25.         try {
  26.             inputStream = new FileInputStream("./test.txt");
  27.         }finally{
  28.             inputStream.close();
  29.         }
  30.         */
  31.         
  32.         
  33.         /*
  34.         //第三种写法,try with resources版本
  35.         try (InputStream inputStreeam = new FileInputStream("./test.txt")){
  36.             //出了try{}代码块范围后,编译器会自动调用.close方法关闭文件
  37.             //FileInputStream继承于InputStream,InputStream又实现了Closeable接口,必须实现了Closeable接口才可以被放到try()里面
  38.         }
  39.         */
  40.         
  41.         
  42.         
  43.     }
  44. }
复制代码

(1)构造方法


注:你给cpu送的信息叫输入,好比读利用。cpu给存储器等这些硬件设备送的信息叫做输出,好比写利用
(2)类/非常解析



(3)文件资源泄漏

如果不实行文件关闭.close方法,就会有大问题。这里本质上是释放了文件的相关资源(PCB文件描述符表,不懂得铁子可以看前面写过的文章)这个表本质是一个数组或者次序表,每打开一个文件,就会占据一个位置,如果不关闭不停打开就会导致表被耗尽,后续再想打开就会报错。
那么提问,文件描述符表不会动态扩容吗?对利用系统的内核来说,需要很高的性能,付出的代价太大了,容易造成系统的卡顿


(4):read方法

①引入

read读文件就是把硬盘数据读取到内存当中buffer,译为缓冲区

留意点①:offset此处缩写了
留意点②:传入的数组,在方法内部对数组内容修改事后,方法外部也能生效
留意点③:字符流读取文件的时候,
②三种参数代码示例

①*不带参数的读,一个一个字符的读
  1. import java.io.FileInputStream;
  2. import java.io.FileNotFoundException;
  3. import java.io.IOException;
  4. import java.io.InputStream;
  5. /**
  6. * Created with IntelliJ IDEA.
  7. * Description:
  8. * User: Hua YY
  9. * Date: 2024-10-02
  10. * Time: 12:18
  11. */
  12. public class IOStream8 {
  13.     public static void main(String[] args) throws IOException {
  14.         try(InputStream inputStream = new FileInputStream("./test.txt")){
  15.             //读取操作,读到-1的时候就会停止
  16.             while(true){
  17.                 int a = inputStream.read();
  18.                 if (a == -1){
  19.                     break;
  20.                 }
  21.                 System.out.print(a + " ");//读取的abcdef是ascii码值
  22.             }
  23.         }
  24.     }
  25. }
复制代码

②*带入数组参数的读取,一次读取多个字符
  1. import java.io.FileInputStream;
  2. import java.io.FileNotFoundException;
  3. import java.io.IOException;
  4. import java.io.InputStream;
  5. /**
  6. * Created with IntelliJ IDEA.
  7. * Description:
  8. * User: Hua YY
  9. * Date: 2024-10-02
  10. * Time: 12:56
  11. */
  12. public class IODemon9 {
  13.     public static void main(String[] args) throws IOException {
  14.         try(InputStream inputStream = new FileInputStream("./test.txt")){
  15.             while (true){
  16.                 //也可以一次读若干个字节,buffer(缓冲区)
  17.                 byte[] buffer = new byte[1024];//读的时候会尝试把buffer[1024]填满
  18.                 int n = inputStream.read(buffer);//传入的参数是buffer数组,方法内部对数组内容进行修改后,方法外部也能生效,实际上填不满1024个字节,
  19.                 if (n == -1){
  20.                     break;//读取到最后一个字符,就break
  21.                 }
  22.                 for (int i = 0; i < n; i++) {
  23.                     System.out.printf(" %x " ,buffer[i]);//遍历数组,并打印
  24.                 }
  25.             }
  26.         }
  27.     }
  28. }
复制代码


③*构造String从0读到n(利用offset)
offset翻译为抵消补偿
留意看while循环:什么是文件流,假设我们要读取的文件很大,在这个循环中,每次尽大概读取1024个字节(大概读到的字节数比这个要小),这次读不完,我们就先用String给存起来(0到n),下一次while循环就接着读,“特别像字符串拼接   ”
  1. import java.io.FileInputStream;
  2. import java.io.IOException;
  3. import java.io.InputStream;
  4. import java.util.Arrays;
  5. /**
  6. * Created with IntelliJ IDEA.
  7. * Description:
  8. * User: Hua YY
  9. * Date: 2024-10-02
  10. * Time: 13:22
  11. */
  12. public class IODemon10 {
  13.     public static void main(String[] args) throws IOException {
  14.         try(InputStream inputStream = new FileInputStream("./test.txt")){
  15.             while (true){
  16.                 //也可以一次读若干个字节,buffer(缓冲区)
  17.                 byte[] buffer = new byte[1024];//读的时候会尝试把buffer[1024]填满
  18.                 int n = inputStream.read(buffer);//传入的参数是buffer数组,方法内部对数组内容进行修改后,方法外部也能生效
  19.                 if (n == -1){
  20.                     break;
  21.                 }
  22.                 String string = new String(buffer , 0 , n);//构造一个String对象,数组前n个字节来构造
  23.                     System.out.printf(string);//遍历数组,并打印
  24.             }
  25.         }
  26.     }
  27. }
复制代码

2:OutputStream

按照写的方式打开文件

  1. import java.io.FileNotFoundException;
  2. import java.io.FileOutputStream;
  3. import java.io.IOException;
  4. import java.io.OutputStream;
  5. /**
  6. * Created with IntelliJ IDEA.
  7. * Description:
  8. * User: Hua YY
  9. * Date: 2024-10-05
  10. * Time: 12:06
  11. */
  12. public class IODemo11 {
  13.     public static void main(String[] args) throws FileNotFoundException {
  14.         try{
  15.             //打开操作会把原有文件中的内容清空,所以罪魁祸首不是Write操作而是打开操作
  16.             OutputStream outputStream = new FileOutputStream("./test.txt");
  17.             //OutputStream outputStream = new FileOutputStream("./test.txt",true);追加写
  18.             byte[] buffer = new byte[]{97,98,99,100,101};
  19.             outputStream.write(buffer);
  20.         }catch (IOException e) {
  21.             throw new RuntimeException(e);
  22.         }
  23.     }
  24. }
复制代码
3:Scanner联动

Scanner(System.in)中System.in本质就是一个InputStream

  1. import java.io.FileInputStream;
  2. import java.io.FileNotFoundException;
  3. import java.io.IOException;
  4. import java.io.InputStream;
  5. import java.util.Scanner;
  6. /**
  7. * Created with IntelliJ IDEA.
  8. * Description:
  9. * User: Hua YY
  10. * Date: 2024-10-05
  11. * Time: 13:21
  12. */
  13. public class IODemo14 {
  14.     public static void main(String[] args) throws FileNotFoundException {
  15.         try(InputStream inputStream = new FileInputStream("./test.txt")){
  16.             Scanner scanner = new Scanner(inputStream);
  17.             while(scanner.hasNext()){
  18.                 String s = scanner.next();
  19.                 System.out.println(s);
  20.             }
  21.         } catch (IOException e) {
  22.             throw new RuntimeException(e);
  23.         }
  24.     }
  25. }
复制代码


六:字符流

以字符为单位进行读写,比方:用utf8表示汉字,3个字节就是一个汉字,每次读写都得以3个字节为单位来进行读写,不能一次读半个汉字
在用char范例的数组存储汉字时是把3个字节组成的utf8汉字转化为了Unicode,输出为String范例时在把Unicode转化为字节
代表类:Reader——输入,Writer——输出

1:Reader类

代码示例
把byte换为char测试一下,区别在于打印汉字
  1. import java.io.*;
  2. /**
  3. * Created with IntelliJ IDEA.
  4. * Description:
  5. * User: Hua YY
  6. * Date: 2024-10-05
  7. * Time: 12:33
  8. */
  9. public class IODemo12 {
  10.     public static void main(String[] args) {
  11.         try{
  12.             //Reader和Writer的字符流与OutputStream和InputStream相似
  13.             Reader reader= new FileReader("./test.txt");
  14.             while(true){
  15.                 char[] buffer = new char[1024];
  16.                 int n = reader.read(buffer);
  17.                 if (n == -1){
  18.                     break;
  19.                 }
  20.                 String s = new String(buffer , 0 , n);
  21.                 System.out.println(s);
  22.             }
  23.         } catch (IOException e) {
  24.             throw new RuntimeException(e);
  25.          }
  26.     }
  27. }
复制代码

如果文件中有内容换行利用了,那么有大概会有一些内容打印不到

2:Writer类 

记着:要把文件打开利用放到try()的括号里,要不然不会默认实行close关闭文件利用
  1. import java.io.FileWriter;
  2. import java.io.IOException;
  3. import java.io.Writer;
  4. /**
  5. * Created with IntelliJ IDEA.
  6. * Description:
  7. * User: Hua YY
  8. * Date: 2024-10-05
  9. * Time: 13:07
  10. */
  11. public class IODemo13 {
  12.     public static void main(String[] args) {
  13.         try(Writer writer = new FileWriter("./test.txt",true)) {
  14.             String s = "可选择续写操作 或者 清空在写操作";
  15.             writer.write(s);
  16.         } catch (IOException e) {
  17.             throw new RuntimeException(e);
  18.         }
  19.     }
  20. }
复制代码




免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

种地

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表