文件操作和IO

一.文件的概念:

狭义的文件指:硬盘上的文件和目录

广义上的文件指:计算机中的很多软硬件资源

路径:

  1. 绝对路径:以c:d盘符开头的路径,比如c:/Intel/Logs/text.txt

  1. 相对路径:以当前所在的目录为基准,以.或者..开头(.)有时候可以省略,找到指定的路径

文件操作和IO_第1张图片

假设当前的工作目录是c:/MyDrivers,定位到update这个目录,就可以表示成./update

同样是定位到c:/相对路径写作./MyDrivers/update,如果工作目录是c:/MyDrivers,相对路径写作./update,如果工作目录是c:/MyDrivers/tmp,相对路径写作../update,如果工作目录是c:/MyDrivers/tmp/123,相对路径写作../../update。

二.Java对于文件的操作:

  1. 针对文件系统的操作(文件的创建,删除,重命名)

  1. 针对文件内容操作(文件的读和写)

java标准库,提供了一个File这个类

文件操作和IO_第2张图片
public static void main(String[] args)throws IOException {
        File file = new File("c:/intel/Logs/cat.txt");//这个文件不一定是真实存在的
        System.out.println(file.getName());//文件名
        System.out.println(file.getPath());
        System.out.println(file.getAbsolutePath());//文件的绝对路径
        System.out.println(file.getParent());//除去文件名的路径
        System.out.println(file.isFile());//判断是不是文件
        System.out.println(file.isDirectory());//判断是不是目录
        file.createNewFile();//创建这样的一个文件出来
        System.out.println(file.isFile());
        file.delete();//删除改文件
    }
文件操作和IO_第3张图片

 public static void main(String[] args) {
        File file=new File("d:/123/giteedemo/util11/in");//创建1级目录
        file.mkdir();
    }
文件操作和IO_第4张图片
public static void main(String[] args) {
        File file=new File("./in/aaa/bbb");
        file.mkdirs();
    }

三.针对文件内容:使用流对象进行操作

Java标准库的流对象从类型上分为两个大类:

  1. 字节流:

InputStream FileInputStream//把数据从硬盘读取到内存中

OutputStream FileOutputStream//把数据从内存写入到硬盘中

  1. 字符流

Reader FileReader

Writer FileWriter

这些类的使用方式非常固定,核心就是四个操作

  1. 打开文件(构造对象)

  1. 关闭文件(close)

  1. 读文件(read) 针对InputStream/Reader

  1. 写文件(write)针对OutputStream/Writer

文件操作和IO_第5张图片
 public static void main(String[] args)throws IOException {
        InputStream inputStream=new FileInputStream("c:/Intel/text2.txt");
        while(true)
        {
            int a=inputStream.read();

            if(a==-1)//表示读取到文件末尾
            {
                break;
            }
            System.out.println((byte)a);//实际上每次读一个字节我们再强转成byte
        }
        inputStream.close();

    }

read 无参数版本,一次读一个字节

read 一个参数版本,把读到的内容填充到参数的这个字节数组里,只要字节数够每次读取的就是这个数组的大小,假设数组大小是1024,那么要读取的数据字节数够的情况下,一次就读124,如果不够124个字节的话,那么这次读取,剩下多少就读多少

 public static void main(String[] args)throws IOException {
        try(InputStream inputStream=new FileInputStream("c:/MyDrivers/pig.jpg"))
        {
            while(true)
            {
                byte[]buffer=new byte[1024];//每次尽力读1024
                int length=inputStream.read(buffer);//返回每次读取的字节数
                System.out.println(length);
                if(length==-1)
                {
                    break;
                }
                //for(int i=0;i
文件操作和IO_第6张图片

最后读取的时候只有377个字节了,所以只读了377个。

像这种读取方式有什么好处呢?单次IO操作,是要访问硬盘/IO设备,单次操作就是比较消耗时间的,如果频繁进行这样的IO操作,肯定就更耗时了,单次IO时间是一定的,如果能缩短IO的次数,此时就可以提高程序的整体效率了,每次只读一个字节,循环次数很高,read次数也很高,但一次 读1024个字节,循环次数就降低很多了,read次数变少了。

read(三个参数版本类型)和一个参数版本类似,往数组里的一部分区间里填充

read的无参数版本,按理说每次读取的是一个字节,返回一个byte就行了,但是实际上返回的是int,因为除了要表示byte里的(-128-127)这样的情况之外,还需要表示一个特殊情况-1,这个情况表示读取文件结束了(读到文件末尾了)

我们来看一下文件里内容是什么

我们再来看一下打印结果

文件操作和IO_第7张图片

我们看到这些数字其实就是love的ascii码值

当文本内容是兄弟时,我们看一下读取结果

文件操作和IO_第8张图片

这里我们需要注意一件事情:

public static void main(String[] args)throws IOException {
            InputStream inputStream=new FileInputStream("c:/Intel/text2.txt");
        while(true)
        {
            int a=inputStream.read();

            if(a==-1)//表示读取到文件末尾
            {
                break;
            }
            System.out.printf("%x"+" ",(byte)a);//实际上每次读一个字节我们再强转成byte
        }
        inputStream.close();

    }

我们看到每次我们都需要close(),含义是关闭文件,那么这里的关闭文件又起到了什么作用呢?进程->在内核里,使用PCB这样的数据结构来表示进程,一个线程对应一个PCB,一个进程可以对应一个PCB,也可以对应多个,PCB有一个重要的属性,文件描述符表(相当于一个数组)记录了该进程打开了哪些文件(即使一个进程里有多个线程多个PCB,也没关系,共同用一个文件描述符表)

文件操作和IO_第9张图片

文件描述符表里的每个元素都是内核里的一个file_struct对象,这个对象就表示一个打开了的文件,每次打开文件操作,就会在文件描述符表中,把这个信息放进去,每次关闭文件,也就会把这个文件描述符表对应的表项给释放掉,如果没有close,对应的表项,没有及时释放,虽然java有GC,GC操作会在回收这个outputStream对象的时候去完成这个释放操作,但是这个GC不一定及时,如果不手动释放,意味着文件描述符表可能很快就被占满了,(这个数组,不能自动扩容,存在上限)如果占满了以后,后面再次打开文件,就会打开失败。

为了防止我们遗忘掉close,我们采取了一种更好的写法

  try(InputStream inputStream=new FileInputStream("c:/Intel/text2.txt"))
            {
            while(true)
            {
              int a=inputStream.read();

              if(a==-1)//表示读取到文件末尾
              {  
                break;
              }
              System.out.printf("%x"+" ",(byte)a);//实际上每次读一个字节我们再强转成byte
             }
        

        }

我们将内容放到try语句块里,这个写法虽然没有显式的写close,实际上是会执行的,只要try语句快执行完毕,就可以自动执行到close,不是随便拿一个对象放到try()里就能自动释放,需要满足一定的要求。

实现了Closeable接口的类才可以放到try的()被自动关闭,这个接口提供的方法就是close方法。

使用了InputStream来读文件,还可以使用OutputStream来写文件。

 public static void main12(String[] args)throws IOException {

        try (OutputStream outputStream = new FileOutputStream("c:/Intel/text.txt")) {
            outputStream.write('h');
            outputStream.write('e');


        }
    }

我们可以看原本是“兄弟”现在是he.

我们接下来再来看一下Reader,Writer按字符读取和写的用法(适合读文本文件)

我们先来看一下我们text的文件里有啥

public static void main13(String[] args)throws IOException {
        Reader reader=new FileReader("c:/Intel/text.txt");
        while(true)
        {
            int a=reader.read();
            if(a==-1)
            {
                break;
            }
            System.out.println(a);
        }

    }

我们看一下打印结果:

文件操作和IO_第10张图片

这是读文件,我们再来看一下写文件

public static void main17(String[] args)throws IOException {
        try (Writer write = new FileWriter("c:/Intel/text.txt")) {

            write.write("同志再见");
        }
    }

我们将文本里写入“同志再见”

文件操作和IO_第11张图片

新写入的内容会覆盖掉原来的内容。

另外像这种写操作,我们在最后最好是加上flush语句

 write.flush();

也就是这样

public static void main999(String[] args)throws IOException {
        try (Writer write = new FileWriter("c:/Intel/text.txt")) {

            write.write("同志再见");
            write.flush();
        }
    }

因为像这种写操作,其实是先写到缓冲区里,写操作执行完了,内容可能还在缓冲区里,还没有真的进入硬盘,close操作,就会触发缓冲区的刷新(刷新操作,就是把缓冲区里的内容写到硬盘里),除了close方法,还可以通过flush方法,也能起到刷新缓冲区的效果。

Scanner 是搭配流对象进行使用的

  Scanner scaner=new Scanner(System.in);//System.in其实就是一个输入流对象
 public static void main(String[] args)throws IOException {
        try(InputStream inputStream=new FileInputStream("c:/Intel/text.txt"))
        {
            Scanner scanner=new Scanner(inputStream);//从文件里读
              while(scanner.hasNext())
              {
                                  
                String s = scanner.next();
                System.out.println(s);
               }                  
            

        }
    }

文件内容:

文件操作和IO_第12张图片

打印结果:

四.两个小程序的实现

  1. 扫描指定目录,并找到名称中包含指定字符的所有普通文件(不包含目录),并且询问用户是否要删除该文件。

ublic static Scanner scan=new Scanner(System.in);
    public static void main(String[] args) {

        System.out.println("请输入路径");
        String basepath=scan.next();
        File root=new File(basepath);
        if(!root.isDirectory())
        {
            System.out.println("输入有误");
            return;
        }
        System.out.println("请输入你想删除的文件");
        String target=scan.next();
        find(root,target);

    }
    public static void find(File root,String target) {
        //System.out.println(root.getAbsolutePath());
        File[] files = root.listFiles();、//列举出当前目录所包含的东西
        if (files == null) {//如果当前目录为空,就返回
            return;
        }
        for (File file : files) {

            if (file.isDirectory()) {//如果是目录,就继续递归
                find(file, target);
            } else {
                if (file.getName().contains(target)) {//包含指定字符的文件
                    System.out.println("请确认是否要删除:删除请按yes,取消删除请按No");
                    String user = scan.next();
                    if (user.equals("yes")) {
                        file.delete();
                        System.out.println("删除成功");

                    } else {
                        System.out.println("取消删除");

                    }


                }
            }
        }
    }
}
文件操作和IO_第13张图片

  1. 进行普通文件的复制:

思路:把第一个文件按照字节依次读取,把结果写入到另一个 文件中

public static Scanner scan=new Scanner(System.in);
    public static void main(String[] args) {
        System.out.println("请输入想要想要复制的文件:");
        String wen=scan.next();
        File file=new File(wen);
        if(!file.isFile())
        {
            System.out.println("文件不存在");
            return;
        }
        System.out.println("请输入文件复制到的路径");
        String lu=scan.next();
        File filelu=new File(lu);
        if(filelu.isFile())
        {
            System.out.println("您当前输入的目标路径有误");
            return;

        }
        try(InputStream inputStream=new FileInputStream(wen); OutputStream outputStream=new FileOutputStream(lu))
        {
            while(true)
            {
                int a=inputStream.read();
                if(a==-1)
                {
                    break;

                }
                outputStream.write(a);
            }
            System.out.println("复制文件成功");

        }catch(IOException e)
        {
            e.printStackTrace();
        }


    }
}
文件操作和IO_第14张图片

你可能感兴趣的:(java,开发语言)