Java学习---Day16_IO流基础篇

Java学习—Day16_IO流基础篇

文件操作

操作磁盘上的某一个文件或某一个文件夹,可以对他们进行创建或删除、移动、属性获取、属性设置等操作。但是,不包含读取文件的内容、拷贝文件

ps:java中,使用Java.io.File类描述一个文件,或者是一个文件夹

绝对路径和相对路径

路径:用来描述一个文件夹所在的地址,用来定位一个文件的,可以分为绝对路径相对路径

  • **绝对路径:**从磁盘的根目录开始,一层层的向下查找,直到找到这个文件
    • c:\Users\Iuds\Deskto\集合\assets\map.png
  • 相对路径:找到一个参照物,相对于这个参照物的路径
    • assets/collection.png
对比
优点
绝对路径用来表示一个文件的地址, 只要这个使用方还在当前磁盘上, 一定可以找到这个文件。一旦更换一个文件系统, 此时这个路径表示的文件将无法找到
相对路径只要两者的相对位置不变, 无论在哪一个文件系统中, 都可以找到这个文件。只要两者的相对位置发生了改变, 这个文件就无法被访问到。

分隔符

在描述路径的字符串总,有里两个分隔符,是比较常见的:目录分隔符路径分隔符

目录分隔符

隔开一个路径中的不同的文件夹, 用来描述层级关系、 包含关系。在不同的操作系统中, 目录分隔符是不一样的。 在windows中, 使用 \ 作为目录分隔符; 在非windows的操作系统中, 例如: Linux、Unix… 使用的是 / 作为目录分隔符

路径分隔符

分隔开一个字符串中的多个路径的。在不同的操作系统中, 路径分隔符是不一样的。 在windows中, 使用 ; 作为路径
分隔符; 在非windows的操作系统中, 例如: Linux、Unix… 使用的是 :
作为路径分隔符

分隔符的表示

如果你的程序只需要考虑部署在windows平台, 那么只需要按照windows的规范书写就可以; 如果你的程序只需要部署到linux上, 那么只需要按照linux的规范书写就可以。 但是, 如果你的程序需要考虑在不同的平台上部署运行, 此时就需要使用以
下方法进行分隔符的获取

方法描述
File.separator();获取一个目录分隔符。 会根据不同的操作系统,返回一个指定的目录分隔符字符串
File.separatorChar()取一个目录分隔符。 会根据不同的操作系统,返回一个指定的目录分隔符字符。
File.pathSeparator()获取一个路径分隔符。 会根据不同的操作系统,返回一个指定的路径分隔符字符串。
File.pathSeparatorChar();获取一个路径分隔符。 会根据不同的操作系统,返回一个指定的路径分隔符字符。

File类

File类的构造方法
import java.io.File;
public class Program {
    public static void main(String[] args) {
        // 1. File(String pathname)
        //   如果这个路径下的文件不存在,不影响File对象的实例化。
        File file = new File("C:\\Users\\luds\\Desktop\\dis_hash.png");
        System.out.println(file.exists());
 
        // 2. File(String parent, String child)
        //    在这个构造方法中,会将parent与child合并在一起
        File file1 = new File("C:\\Users\\luds\\Desktop","dis_hash.png");
        System.out.println(file1);
        System.out.println(file1.exists());
        
        // 3. File(File parent, String child)
        //    在构造方法中,将parent的路径和child进行拼接,得到一个新的路径
        File desktop = new File("C:\\Users\\luds\\Desktop");
        File file2 = new File(desktop, "dis_hash.png");
        System.out.println(file2);
        System.out.println(file2.exists());
    }
}
File类的常用方法
  1. 文件属性获取方法
import java.io.File;
import java.util.Date;
public class FileOperation {
    public static void main(String[] args) {
        // 实例化一个对象
        //1.直接通过绝对路径
        //File file1 = new File("D:\\BigData2005\\BigData2005N19\\src\\com\\qf\\test\\Demo1.java");
//      //2.通过父路径和子路径字符串形式的拼接
        //File file2 = new File("D:\\BigData2005\\BigData2005N19\\","src\\com\\qf\\test\\Demo1.java");
//      //3.通过父路径对象和子路径对象(字符串)
//        File file3 = new File("D:\\BigData2005\\BigData2005N19\\");
//        File file4 = new File(file3,"src\\com\\qf\\test\\Demo1.java");
      
        
      //实例演示
        File file = new File("src\\day24\\cFiles\\FileOperation.java");
 
        // 1. 判断一个File对象指向的路径上是否有文件或者文件夹
        System.out.println("exists = " + file.exists());
        // 2. 判断一个File对象指向的路径上是否是一个文件
        System.out.println("isFile = " + file.isFile());
        // 3. 判断一个File对象指向的路径上是否是一个目录
        System.out.println("isDir = " + file.isDirectory());
        // 4. 获取一个文件的大小(注:只能获取文件的大小,不能获取文件夹的大小)
        System.out.println("length = " + file.length());
        // 5. 判断一个文件(目录)是否是隐藏的
        System.out.println("hidden = " + file.isHidden());
        // 6. 判断文件(目录)的权限
        System.out.println("read = " + file.canRead());         
// 可读权限
        System.out.println("write = " + file.canWrite());       
// 可写权限
        System.out.println("execute = " + file.canExecute());   
// 可执行权限
        // 7. 获取文件的名字
        System.out.println("name = " + file.getName());
        // 8. 获取文件的路径
        System.out.println("path = " + file.getPath());         
        // 相对路径
        System.out.println("absolutePath = " + file.getAbsolutePath()); // 绝对路径
        // 9. 获取父级文件夹的路径(字符串)
        System.out.println("parent = " + file.getParent());
        // 10. 获取父级文件夹(File对象)
        System.out.println("parentFile = " + file.getParentFile());
        // 11. 获取文件上次修改的时间(时间戳)--是最后修改的时间,不是查看的时间
//        long lastTime = file.lastModified();
//        System.out.println(lastTime);
//        Date date = new Date(lastTime);
//        SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
//        String strDate = simpleDateFormat.format(date);
//        System.out.println(strDate);
 
    }
}
  1. 文件操作方法
import java.io.File;
import java.io.IOException;
public class FileOperation2 {
    public static void main(String[] args) {
        // 实例化一个File对象
        File file = new File("C:\\Users\\luds\\Desktop\\abc\\a\\b\\c");
 
        try {
            // 1. 创建文件
        /*
         * 注意点:
         * 1.必须保证文件以外的路径都是存在的
         * 2.createNewFile只能用于创建文件
         */
            boolean flag = file.createNewFile();
            System.out.println(flag);
        } catch (IOException e) {
            e.printStackTrace();
        }
 
        // 2. 创建文件夹(只能创建一级文件夹)
        boolean flag = file.mkdir();
         System.out.println(flag);
 
        // 3. 创建文件夹(可以创建一级及以上的文件夹)
        boolean flag = file.mkdirs();
        System.out.println(flag);
 
        // 4. 删除文件(目录)(可以删除文件,也可以删除空文件夹)
        //    谨慎使用: 这个方法,可以将文件直接从磁盘删除,不经过回收站。没有撤销的余地。
        boolean flag = file.delete();
        System.out.println(flag);
 
        // 5. 文件的重命名
        File src = new File("C:\\Users\\luds\\Desktop\\abc");
        File dst = new File("C:\\Users\\luds\\Desktop\\ABC");
        System.out.println(src.renameTo(dst));
        // 5.1. 借助重命名,实现文件的移动
        File src1 = new File("C:\\Users\\luds\\Desktop\\DIS_HASH.png");
        File dst1 = new  File("C:\\Users\\luds\\Desktop\\ABC\\dis_hash.png");

        System.out.println(src1.renameTo(dst1));
    }
}
  1. 子文件获取
import java.io.File;
public class FileOperation4 {
    public static void main(String[] args) {
        // 实例化一个File对象
        File file = new File("C:\\Users\\luds\\Desktop");
        // 1. 列举一个目录下所有的子文件的名字
        String[] names = file.list();
        for (String name : names) {
            System.out.println(name);
        }
 
        // 2. 列举一个目录下所有的子文件的名字
        //    带有过滤信息的。
        String[] names1 = file.list((f, name) -> {
            // f: 父级文件夹的File对象
            // name: 子文件的名字
            // 返回值: 如果是true,将会在结果的数组中展示
            return name.startsWith(".");
        });
        for (String s : names1) {
            System.out.println(s);
        }
 
        // 3. 列举一个目录下所有的子文件(以File对象的方式)
        File[] files = file.listFiles();
        for (File file1 : files) {
            System.out.println(file1);
        }
 
        // 4. 列举一个目录下所有的满足指定条件的子文件
        File[] files1 = file.listFiles(File::isHidden);
        for (File file1 : files1) {
            System.out.println(file1);
        }
 
        // 5. 举一个目录下所有的满足指定条件的子文件
        File[] files2 = file.listFiles((f, n) -> new File(f,n).isHidden());
    }
}

IO流

概念及作用

IO流(定义):Input/Output Stream

:指的是一串流动的数据,数据在流中按照指定的方向进行流动,实现数据的读取,写入的功能。

作用:实现两设备之间的数传递

使用场景

**背景:**使用File类, 只能做关于文件的操作, 获取属性、 创建文件、 删除文件、 移动文件等操作, 但是不包含读取文件中的内容。 如果需要读取、修改文件中的内容, 此时就需要使用IO流来完成了

**使用场景:**对一个文件进行读取或写入操作

ps:IO流是对一个文件进行读写的,不是一个文件夹,在使用IO流的时候,不要建立与一个文件夹的连接

设备

设备概念:能输出或者输入数据的都可以成为设备

设备:磁盘(硬盘),内存,键盘,文件,网络,控制台

网络:当前主机之外的网上资源

IO流的分类

按照不同的分类标准, 能够得到不同分类的IO流

  • 按照传输数据的单位:
    • 字节流: 传输的是字节,是以字节为单位的。可以操作任意类型的数据 ------
      音频,视频,文件,图片等
    • 字符流: 传输的是字节,不同点是在传输过程中加入了编码的操作,让我们的操
      作更方便------文本
  • 按照数据传输的方向(以内存为参考):
    • 输入流: 数据从其他设备传到内存(读入)
    • 输出流: 数据从内存传到其他设备(写出)

java.io 包中, 有很多很多的类, 都是来描述IO流的。 但是基本上所有的IO流的类, 都是直接或间接的继承自四大父类流

字节流的两个父类:

  • 字节输入流:InputStream
  • 字节输出流:OutputStream

字符流的两个父类:

  • 字符读入流:Reader
  • 字符写出流:Write

ps

  • 四大父类流, 都是抽象类, 都不能实例化对象。 因此, 需要借助他们的子类实现数据的读写。
  • 流对象一旦实例化完成, 将建立一个程序与文件之间的连接。 这个连接会持有这个文件。 如果这个连接不断, 此时这个文件就是一个被使用中的状态, 此时将无法对这个文件进行其他的操作, 例如删除。
  • 一个流在使用完成之后, 切记! 一定要进行流的关闭。
建立程序与文件的连接

其实, 就是建立了程序与文件之间连接的管道, 实现数据在这个管道之内进行流动。 管道分为不同的类型: 字节输入流、 字节输出流、 字符输入流、 字符输出流。 下面以字节输入流 InputStream 为例。

标准流程
  • try结构外面, 声明流对象, 为了在finally中使用。
  • 在try结构里面, 实例化流对象, 并捕获异常。
  • 在finally结构中, 对流进行关闭。 在关闭的时候, 需要考虑流对象是否是null, 以及要处理 IOException 异常。
import java.io.*;
public class IO1 {
    public static void main(String[] args) {
        // 在外面声明变量
        InputStream inputStream = null;
        try {
            // 实例化一个FileInputStream对象,向上转型为InputStream
类型。
            // 这个实例化如果完成,将会建立程序与文件之间的连接。
            // 建立好之后,数据就可以从文件中流动到程序中。
        // 注意: 一:数据流动到程序中,并不意味着文件中没有数据了!
        //二:如果只写文件的相对路径,不写绝对路径,默认路径是当前的工程
        //FileNotFoundException:(系统找不到指定的路径。)
            inputStream = new FileInputStream("t");
            // 数据的读取操作
            // 在数据读取的过程中,也会出现 IOException 异常。一旦出现异常,后序的代码都不执行了,直接执行catch语句了
            // 流的关闭,不能放到try里面。需要放到finally中。
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } finally {
            // 流在使用结束之后,一定要进行关闭。
            if (inputStream != null) {
                try {
                    inputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}
try结构的特殊使用

JDK1.7 之后, 可以在try后面添加一对小括号。 将 AutoClosable 接口实现类的对象, 实例化放到小括号中完成。 此时, 在try结构执行结束的时候, 会自动的调用AutoClosable接口实现类中的close方法, 进行流的关闭。 这样写的流的建立比较简单, 也是后面我们最主要使用的方式。

import java.io.*;
public class IO2 {
    public static void main(String[] args) {
        /**
         * try结构的特殊语法: try ()
         * 将 AutoClosable 接口的实现类对象的实例化放到小括号中。
         * 此时,在离开了try结构的时候,会自动的对这个类进行close方法的调用
         */
        try (InputStream inputStream = new 
FileInputStream("file\\day25\\source")) {
        // 数据的读取操作
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        System.out.println(new 
File("file\\day25\\source").delete());
    }
}
InputStream

这是一个字节输入流。 从方向来说, 是一个输入流, 数据是从文件中流动到程序中(数据从其他设备到内存), 是为了读取文件中的数据的。 从数据单位来说, 这个流中流动的数据直接是字节的形式。

文件的读取

注意:

1.为了方便测试,我们可以先通过字节输出流或者手动在当前工程中创建一个test1.txt文件,写入字符串abcde
2.有三种读取数据的方式:**第一:**一次读取一个字节;**第二:**一次读取多个字节;**第三:**一次读取全部字节
三种读取数据的方式中,推荐使用第二种,一次读取多个字节

1) 一次读取一个字节

read():一个字节一个字节的读,每次读出一个字节

原理:控制磁头每次向后移动一个字节,依次读取,直到将所有的字节读完啊。当read()返回-1的时候,说明读完了

 public static void read1() throws IOException {
        // 1. 建立程序与文件之间的连接,用来读取这个文件
        try(InputStream inputStream = new 
FileInputStream("test1.txt")){
            //2.读取数据,先声明一个变量,read方法的返回值,就是读取的数据
            //注意:返回-1,默认认为数据读完了
            int num = 0;
//        num = inputStream.read();
            //3.直接读取数据
            //想显示字符需要自己强转
//        System.out.println((char)num);
//        num = inputStream.read();
//        System.out.println((char)num);
//        num = inputStream.read();
//        System.out.println((char)num);
//        num = inputStream.read();
//        System.out.println((char)num);
//        num = inputStream.read();
//        System.out.println((char)num);
//        num = inputStream.read();
//        System.out.println(num);//返回-1,默认认为数据读完了
 
            //4.使用循环读取数据
            while ((num = inputStream.read()) != -1) {
                System.out.print((char)num);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

2) 一次读取多个字节

read(数组):一次可以读出多个字节,数组的作用:每次会将读出的字节临时放到这个数组中

ps:按照数组读,本质上也是一个一个的读

public static void read2() throws IOException {
        // 1. 建立程序与文件之间的连接,用来读取这个文件
        try (InputStream inputStream = new 
FileInputStream("test1.txt")) {
            // 2. 读取字节流中的数据,需要有一个字节数组,用来读取数据
            //创建临时数组
            /* 数组是临时存放数据的地方,我们会将读到的字符放到临时数组中,
数组的大小决定了我们一次可以读到的字符个数.
             * 一般这个数组的大小<=1kB
             * 返回值代表本次读到的真实的字符个数,如果返回值是-1代表读完
了.
             */
            byte[] arr = new byte[2];
            // 3. 声明一个整型变量,用来记录每次读取了多少个字节的数据
            int num = 0;
            //4.直接读取数据
//        num = reader.read(arr);
//        System.out.println(new String(arr,0,num)+"    num:"+num);
//        num = reader.read(arr);
//        System.out.println(new String(arr,0,num)+"    num:"+num);
//        num = reader.read(arr);
            //转部分字符到字符串,第二个参数是指定的下标,第三参数是字符数// 将读取到的字节数组中的数据,转成字符串输出
            // 为了去除最后一次进行读取数据的时候,上次读取残留的问题
            // 最后一次读取的数据,只有指定部分是我们需要的数据
//        System.out.println(new String(arr,0,num)+"    
num:"+num);
//        num = reader.read(arr);
//        System.out.println("    num:"+num);
 
            //5.使用循环读取数据
            while ((num = inputStream.read(arr)) != -1){
                System.out.println(new String(arr,0,num)+"    num:"+num);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

3) 一次读取全部字节

可以通过available()方法获取全部字节个数

public static void read3() throws IOException {
        // 1. 建立程序与文件之间的连接,用来读取这个文件
        try (InputStream inputStream = new 
FileInputStream("test1.txt")) {
            // 2. 读取字节流中的数据,需要有一个字节数组,用来读取数据
            //获取文件的字节个数
            //注意:这种方式适合文件的字节数比较小的时候,大概是几kb之内.
            int num = inputStream.available();
            //2.读
            byte[] bytes = new byte[num];
            inputStream.read(bytes);
            System.out.println(new String(bytes));
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
OutputStream

字节输出流。 从方向上来分, 是一个输出流, 数据从程序中流动到文件中(数据从内存到其他设备), 实现文件的写操作。 从流中流动的数据单位来分, 是一个字节流,流中流动的数据直接是字节的形式。

import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
public class OutputStreamTest {
    public static void main(String[] args) {
        // 1. 实例化一个管道,连接文件和程序。
        //    对于FileOutputStream来说,如果目标文件不存在,则会自动的创建,如果存在,会将原来的内容覆盖
        //    当无法创建这个文件的时候(父级目录不存在),创建会失败,会触发 FileNotFoundException。
        //OutputStream outputStream = new FileOutputStream("test1.txt")
        //文件的续写:FileWriter(String file,boolean value)
        //当value位true的时候,不会将原来的内容覆盖,会接着写
        try (OutputStream outputStream = new 
FileOutputStream("test1.txt",true)) {
            // 2. 准备需要写入到这个文件中的数据
            String message = "你好,师姐";
            // 3. 将数据写入到输出流中,由输出流写入到文件中
            //将字符串转成字节数组
            outputStream.write(message.getBytes());
 
            // 冲刷缓冲区,将缓冲区中的数据强制流动到文件中。
            // 在流关闭的时候,会自动的调用。
            //注意:当我们进行循环写入操作时,最好通过flush()方法刷新
            outputStream.flush();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}
Reader

这是一个字符输入流。 从方向来说, 是一个输入流, 数据是从文件中流动到程序中(数据从其他设备到内存), 是为了读取文件中的数据的。 从数据单位来说, 这个流中流动的数据以字节为单位的,不同的是在传输过程中加入了编码的操作,让我们的操作更方便。

import java.io.FileReader;
import java.io.IOException;
import java.io.Reader;
public class ReaderTest {
    public static void main(String[] args) {
        // 读取过程与字节输入流完全相同,只需要将使用到的类换一下即可。
        try (Reader reader = new FileReader("file\\day25\\src")) 
{
            // 1. 实例化一个字符数组
            char[] array = new char[100];
            // 2. 声明一个变量,用来记录每次读取到了多少个数据
            int length = 0;
            // 3. 循环读取数据
            while ((length = reader.read(array)) != -1) {
                String str = new String(array, 0, length);
                System.out.print(str);
            }
        }
        catch (IOException e) {
            e.printStackTrace();
        }
    }
}
Writer

字符输出流。 从方向上来分, 是一个输出流, 数据从程序中流动到文件中(数据从内存到其他设备), 实现文件的写操作。 从流中流动的数据单位来分, 是一个字符流,流中流动的数据是以字节为单位,不同的是在传输过程中加入了编码的操作,让我们的操作更方便。

import java.io.FileWriter;
import java.io.IOException;
import java.io.Writer;
public class WriterTest {
    public static void main(String[] args) {
        // 1. 实例化相关的类
        try (Writer writer = new 
FileWriter("file\\day25\\target", true)) {
            // 2. 将数据写入到输出流中
            writer.write("hello, world");
            // 3. 冲刷缓冲区
            writer.flush();
        }
        catch (IOException e) {
            e.printStackTrace();
        }
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值