JAVA基础系列(七) IO流

作者: Acamy丶 | 来源:发表于2017-06-02 20:01 被阅读68次

    IO流体系分字节流和字符流,里面又都分为输入输出流。下图展示了IO体系的框架图。粗体部分为本文涉及到的IO流。

    本文主要是采用输入输出配对方式进行IO流的讲解,同时附带有相关的类的介绍,本文内容构架如下图:

    1. File

    文件和目录路径名的抽象表示形式

    1.1 构造方法

    public File(String pathname)
    public File(String parent,String child)
    public File(File parent,String child)
    
    

    1.2 成员方法

    创建功能:
    public boolean createNewFile()//创建文件 如果存在这样的文件,就不创建了
    public boolean mkdir()//创建文件夹 如果存在这样的文件夹,就不创建了
    public boolean mkdirs()//创建文件夹,如果父文件夹不存在,会帮你创建出来
    
    删除功能:
    public boolean delete()
    注意:
    A:如果你创建文件或者文件夹忘了写盘符路径,那么,默认在项目路径下。
    B:Java中的删除不走回收站。
    C:要删除一个文件夹,请注意该文件夹内不能包含文件或者文件夹
    
    重命名功能:
    public boolean renameTo(File dest)
    注意:
    如果路径名相同,就是改名。
    如果路径名不同,就是改名并剪切。
    
    判断功能:
    public boolean isDirectory()//判断是否是目录
    public boolean isFile()//判断是否是文件
    public boolean exists()//判断是否存在
    public boolean canRead()//判断是否可读
    public boolean canWrite()//判断是否可写
    public boolean isHidden()//判断是否隐藏
    
    获取功能:
    public String getAbsolutePath():获取绝对路径
    public String getPath():获取相对路径
    public String getName():获取名称
    public long length():获取长度。字节数
    public long lastModified():获取最后一次的修改时间,毫秒值
    public String[] list():获取指定目录下的所有文件或者文件夹的名称数组
    public File[] listFiles():获取指定目录下的所有文件或者文件夹的File数组
    
    

    ** Demo:**

    public class FileDemo {
        public static void main(String[] args) {
            // 封装e判断目录
            File file = new File("e:\\");
    
            // 获取该目录下所有文件或者文件夹的File数组
            File[] fileArray = file.listFiles();
    
            // 遍历该File数组,得到每一个File对象,然后判断
            for (File f : fileArray) {
                // 是否是文件
                if (f.isFile()) {
                    // 继续判断是否以.jpg结尾
                    if (f.getName().endsWith(".jpg")) {
                        // 就输出该文件名称
                        System.out.println(f.getName());
                    }
                }
            }
        }
    }
    

    2.字节流

    2.1 字节流写数据:FileOutputStream

    字节输出流操作步骤:

    • A:创建字节输出流对象
    • B:调用write()方法
    • C:释放资源

    构造方法

    FileOutputStream(File file) 
    FileOutputStream(String name)
    FileOutputStream(String name, boolean append)
    

    写数据的方式

    public void write(int b):写一个字节
    public void write(byte[] b):写一个字节数组
    public void write(byte[] b,int off,int len):写一个字节数组的一部分
    

    ** Demo:**

    public class FileOutputStreamDemo {
        public static void main(String[] args) throws IOException {
            // 创建字节输出流对象
            // FileOutputStream(File file)
            // File file = new File("fos.txt");
            // FileOutputStream fos = new FileOutputStream(file);
            // FileOutputStream(String name)
            FileOutputStream fos = new FileOutputStream("fos.txt");
            /*
             * 创建字节输出流对象了做了几件事情: A:调用系统功能去创建文件 B:创建fos对象 C:把fos对象指向这个文件
             */
    
            // 写数据
            fos.write("hello,IO".getBytes());
            fos.write("java".getBytes());
    
            // 释放资源
            // 关闭此文件输出流并释放与此流有关的所有系统资源。
            fos.close();
            /*
             * 为什么一定要close()呢? A:让流对象变成垃圾,这样就可以被垃圾回收器回收了 B:通知系统去释放跟该文件相关的资源
             */
            // java.io.IOException: Stream Closed
            // fos.write("java".getBytes());
        }
    }
    

    2.2 字节流读数据:FileInputStream

    字节输入流操作步骤:

    • A:创建字节输入流对象
    • B:调用read()方法读取数据,并把数据显示在控制台
    • C:释放资源

    构造方法:

    FileInputStream(File file)
    FileInputStream(String name)
    

    读取数据的方式

    A:int read():一次读取一个字节
    B:int read(byte[] b):一次读取一个字节数组
    

    ** Demo:**

    public class FileInputStreamDemo2 {
        public static void main(String[] args) throws IOException {
            // 创建字节输入流对象
            // FileInputStream fis = new FileInputStream("fis2.txt");
            FileInputStream fis = new FileInputStream("FileOutputStreamDemo.java");
    
            // 读取数据
            // 定义一个字节数组
            // 第一次读取
            // byte[] bys = new byte[5];
            // int len = fis.read(bys);
            // // System.out.println(len);
            // // System.out.println(new String(bys));
            // // System.out.println(new String(bys, 0, len));
            // System.out.print(new String(bys, 0, len));
            //
            // // 第二次读取
            // len = fis.read(bys);
            // // System.out.println(len);
            // // System.out.println(new String(bys));
            // // System.out.println(new String(bys, 0, len));
            // System.out.print(new String(bys, 0, len));
            //
            // // 第三次读取
            // len = fis.read(bys);
            // // System.out.println(len);
            // // System.out.println(new String(bys));
            // // System.out.println(new String(bys, 0, len));
            // System.out.print(new String(bys, 0, len));
            //
            // // 第四次读取
            // len = fis.read(bys);
            // // System.out.println(len);
            // // System.out.println(new String(bys, 0, len));
            // System.out.print(new String(bys, 0, len));
            // // 代码重复了,用循环改进
            // // 但是,我不知道结束条件
            // // len = fis.read(bys);
            // // System.out.println(len);
            // // len = fis.read(bys);
            // // System.out.println(len);
            // // 如果读取到的实际长度是-1,就说明没有数据了
    
            // byte[] bys = new byte[115]; // 0
            // int len = 0;
            // while ((len = fis.read(bys)) != -1) {
            // System.out.print(new String(bys, 0, len));
            // // System.out.print(new String(bys)); //千万要带上len的使用
            // }
    
            // 最终版代码
            // 数组的长度一般是1024或者1024的整数倍
            byte[] bys = new byte[1024];
            int len = 0;
            while ((len = fis.read(bys)) != -1) {
                System.out.print(new String(bys, 0, len));
            }
    
            // 释放资源
            fis.close();
        }
    }
    

    3. 缓冲区流(高效流)

    3.1 写数据:BufferedOutputStream

    ** Demo:**

    public class BufferedOutputStreamDemo {
        public static void main(String[] args) throws IOException {
            // BufferedOutputStream(OutputStream out)
            // FileOutputStream fos = new FileOutputStream("bos.txt");
            // BufferedOutputStream bos = new BufferedOutputStream(fos);
            // 简单写法
            BufferedOutputStream bos = new BufferedOutputStream(
                    new FileOutputStream("bos.txt"));
    
            // 写数据
            bos.write("hello".getBytes());
    
            // 释放资源
            bos.close();
        }
    }
    

    3.2 读数据:BufferedInputStream

    ** Demo:**

    public class BufferedInputStreamDemo {
        public static void main(String[] args) throws IOException {
            // BufferedInputStream(InputStream in)
            BufferedInputStream bis = new BufferedInputStream(new FileInputStream(
                    "bos.txt"));
    
            // 读取数据
            // int by = 0;
            // while ((by = bis.read()) != -1) {
            // System.out.print((char) by);
            // }
            // System.out.println("---------");
    
            byte[] bys = new byte[1024];
            int len = 0;
            while ((len = bis.read(bys)) != -1) {
                System.out.print(new String(bys, 0, len));
            }
    
            // 释放资源
            bis.close();
        }
    }
    

    字节流和缓冲流复制文件的对比Demo:

    import java.io.BufferedInputStream;
    import java.io.BufferedOutputStream;
    import java.io.FileInputStream;
    import java.io.FileOutputStream;
    import java.io.IOException;
    
    /*
     * 需求:把e:\\哥有老婆.mp4复制到当前项目目录下的copy.mp4中
     * 
     * 字节流四种方式复制文件:
     * 基本字节流一次读写一个字节:   共耗时:117235毫秒
     * 基本字节流一次读写一个字节数组: 共耗时:156毫秒
     * 高效字节流一次读写一个字节: 共耗时:1141毫秒
     * 高效字节流一次读写一个字节数组: 共耗时:47毫秒
     */
    public class CopyMp4Demo {
        public static void main(String[] args) throws IOException {
            long start = System.currentTimeMillis();
            method4("e:\\哥有老婆.mp4", "copy4.mp4");
            long end = System.currentTimeMillis();
            System.out.println("共耗时:" + (end - start) + "毫秒");
        }
    
        // 高效字节流一次读写一个字节数组:
        public static void method4(String srcString, String destString)
                throws IOException {
            BufferedInputStream bis = new BufferedInputStream(new FileInputStream(
                    srcString));
            BufferedOutputStream bos = new BufferedOutputStream(
                    new FileOutputStream(destString));
    
            byte[] bys = new byte[1024];
            int len = 0;
            while ((len = bis.read(bys)) != -1) {
                bos.write(bys, 0, len);
            }
    
            bos.close();
            bis.close();
        }
    
        // 高效字节流一次读写一个字节:
        public static void method3(String srcString, String destString)
                throws IOException {
            BufferedInputStream bis = new BufferedInputStream(new FileInputStream(
                    srcString));
            BufferedOutputStream bos = new BufferedOutputStream(
                    new FileOutputStream(destString));
    
            int by = 0;
            while ((by = bis.read()) != -1) {
                bos.write(by);
    
            }
    
            bos.close();
            bis.close();
        }
    
        // 基本字节流一次读写一个字节数组
        public static void method2(String srcString, String destString)
                throws IOException {
            FileInputStream fis = new FileInputStream(srcString);
            FileOutputStream fos = new FileOutputStream(destString);
    
            byte[] bys = new byte[1024];
            int len = 0;
            while ((len = fis.read(bys)) != -1) {
                fos.write(bys, 0, len);
            }
    
            fos.close();
            fis.close();
        }
    
        // 基本字节流一次读写一个字节
        public static void method1(String srcString, String destString)
                throws IOException {
            FileInputStream fis = new FileInputStream(srcString);
            FileOutputStream fos = new FileOutputStream(destString);
    
            int by = 0;
            while ((by = fis.read()) != -1) {
                fos.write(by);
            }
    
            fos.close();
            fis.close();
        }
    }
    

    4. 转换流

    4.1 OutputStreamWriter

    构造方法

    //根据默认编码把字节流的数据转换为字符流
    OutputStreamWriter(OutputStream out);
    //根据指定编码把字节流数据转换为字符流
    OutputStreamWriter(OutputStream out,String charsetName)
    
    

    写数据方法

    public void write(int c)
    public void write(char[] cbuf)
    public void write(char[] cbuf,int off,int len)
    public void write(String str)
    public void write(String str,int off,int len)
    

    ** Demo:**

    public class OutputStreamWriterDemo {
        public static void main(String[] args) throws IOException {
            // 创建对象
            // OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream(
            // "osw.txt")); // 默认GBK
            // OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream(
            // "osw.txt"), "GBK"); // 指定GBK
            OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream(
                    "osw.txt"), "UTF-8"); // 指定UTF-8
            // 写数据
            osw.write("中国");
    
            // 释放资源
            osw.close();
        }
    }
    

    4.2 InputStreamReader

    构造方法

    //用默认的编码读取数据
    InputStreamReader(InputStream is);
    //用指定的编码读取数据
    InputStreamReader(InputStream is,String charsetName);
    

    读数据方法

    public int read()
    public int read(char[] cbuf)
    

    ** Demo:**

    
    public class InputStreamReaderDemo {
        public static void main(String[] args) throws IOException {
            // 创建对象
            // InputStreamReader isr = new InputStreamReader(new FileInputStream(
            // "osw.txt"));
    
            // InputStreamReader isr = new InputStreamReader(new FileInputStream(
            // "osw.txt"), "GBK");
    
            InputStreamReader isr = new InputStreamReader(new FileInputStream(
                    "osw.txt"), "UTF-8");
    
            // 读取数据
            // 一次读取一个字符
            int ch = 0;
            while ((ch = isr.read()) != -1) {
                System.out.print((char) ch);
            }
    
            // 释放资源
            isr.close();
        }
    }
    
    

    4.3 FileWriter & FileReader

    转换流的名字比较长,而我们常见的操作都是按照本地默认编码实现的,所以,为了简化我们的书写,转换流提供了对应的子类。
    FileWriter
    FileReader

    ** Demo:**

    import java.io.FileReader;
    import java.io.FileWriter;
    import java.io.IOException;
    
    /*
     * 由于我们常见的操作都是使用本地默认编码,所以,不用指定编码。
     * 而转换流的名称有点长,所以,Java就提供了其子类供我们使用。
     * OutputStreamWriter = FileOutputStream + 编码表(GBK)
     * FileWriter = FileOutputStream + 编码表(GBK)
     * 
     * InputStreamReader = FileInputStream + 编码表(GBK)
     * FileReader = FileInputStream + 编码表(GBK)
     * 
     /*
     * 需求:把当前项目目录下的a.txt内容复制到当前项目目录下的b.txt中
     * 
     * 数据源:
     *      a.txt -- 读取数据 -- 字符转换流 -- InputStreamReader -- FileReader
     * 目的地:
     *      b.txt -- 写出数据 -- 字符转换流 -- OutputStreamWriter -- FileWriter
     */
    public class CopyFileDemo2 {
        public static void main(String[] args) throws IOException {
            // 封装数据源
            FileReader fr = new FileReader("a.txt");
            // 封装目的地
            FileWriter fw = new FileWriter("b.txt");
    
            // 一次一个字符
            // int ch = 0;
            // while ((ch = fr.read()) != -1) {
            // fw.write(ch);
            // }
    
            // 一次一个字符数组
            char[] chs = new char[1024];
            int len = 0;
            while ((len = fr.read(chs)) != -1) {
                fw.write(chs, 0, len);
                fw.flush();
            }
    
            // 释放资源
            fw.close();
            fr.close();
        }
    }
    

    5.字符缓冲流

    5.1 字符缓冲输出流:BufferedWriter

    字符流为了高效读写,也提供了对应的字符缓冲流。
    将文本写入字符输出流,缓冲各个字符,从而提供单个字符、数组和字符串的高效写入。
    可以指定缓冲区的大小,或者接受默认的大小。在大多数情况下,默认值就足够大了。

    特殊方法

    public void newLine():根据系统来决定换行符

    ** Demo:**

    import java.io.BufferedWriter;
    import java.io.FileWriter;
    import java.io.IOException;
    
    public class BufferedWriterDemo {
        public static void main(String[] args) throws IOException {
            // BufferedWriter(Writer out)
            // BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(
            // new FileOutputStream("bw.txt")));
    
            BufferedWriter bw = new BufferedWriter(new FileWriter("bw.txt"));
    
            bw.write("hello");
            bw.write("world");
            bw.write("java");
            bw.flush();
    
            bw.close();
        }
    }
    
    

    5.2 字符缓冲输入流:BufferedReader

    从字符输入流中读取文本,缓冲各个字符,从而实现字符、数组和行的高效读取。 可以指定缓冲区的大小,或者可使用默认的大小。大多数情况下,默认值就足够大了。

    特殊方法

    public String readLine():一次读取一行数据
    //包含该行内容的字符串,不包含任何行终止符,如果已到达流末尾,则返回 null
    

    ** Demo:**

    import java.io.BufferedReader;
    import java.io.FileReader;
    import java.io.IOException;
    
    public class BufferedReaderDemo {
        public static void main(String[] args) throws IOException {
            // 创建字符缓冲输入流对象
            BufferedReader br = new BufferedReader(new FileReader("bw.txt"));
    
            // 方式1
            // int ch = 0;
            // while ((ch = br.read()) != -1) {
            // System.out.print((char) ch);
            // }
    
            // 方式2
            char[] chs = new char[1024];
            int len = 0;
            while ((len = br.read(chs)) != -1) {
                System.out.print(new String(chs, 0, len));
            }
    
            // 释放资源
            br.close();
        }
    }
    

    6.操作基本数据类型的流

    Demo:

    import java.io.DataInputStream;
    import java.io.DataOutputStream;
    import java.io.FileInputStream;
    import java.io.FileOutputStream;
    import java.io.IOException;
    
    /*
     * 可以读写基本数据类型的数据
     * 数据输入流:DataInputStream
     *          DataInputStream(InputStream in)
     * 数据输出流:DataOutputStream
     *          DataOutputStream(OutputStream out) 
     */
    public class DataStreamDemo {
        public static void main(String[] args) throws IOException {
            // 写
            // write();
    
            // 读
            read();
        }
    
        private static void read() throws IOException {
            // DataInputStream(InputStream in)
            // 创建数据输入流对象
            DataInputStream dis = new DataInputStream(
                    new FileInputStream("dos.txt"));
    
            // 读数据
            byte b = dis.readByte();
            short s = dis.readShort();
            int i = dis.readInt();
            long l = dis.readLong();
            float f = dis.readFloat();
            double d = dis.readDouble();
            char c = dis.readChar();
            boolean bb = dis.readBoolean();
    
            // 释放资源
            dis.close();
    
            System.out.println(b);
            System.out.println(s);
            System.out.println(i);
            System.out.println(l);
            System.out.println(f);
            System.out.println(d);
            System.out.println(c);
            System.out.println(bb);
        }
    
        private static void write() throws IOException {
            // DataOutputStream(OutputStream out)
            // 创建数据输出流对象
            DataOutputStream dos = new DataOutputStream(new FileOutputStream(
                    "dos.txt"));
    
            // 写数据了
            dos.writeByte(10);
            dos.writeShort(100);
            dos.writeInt(1000);
            dos.writeLong(10000);
            dos.writeFloat(12.34F);
            dos.writeDouble(12.56);
            dos.writeChar('a');
            dos.writeBoolean(true);
    
            // 释放资源
            dos.close();
        }
    }
    

    7.内存操作流

    Demo:

    import java.io.ByteArrayInputStream;
    import java.io.ByteArrayOutputStream;
    import java.io.IOException;
    
    /*
     * 内存操作流:用于处理临时存储信息的,程序结束,数据就从内存中消失。
     * 字节数组:
     *      ByteArrayInputStream
     *      ByteArrayOutputStream
     * 字符数组:
     *      CharArrayReader
     *      CharArrayWriter
     * 字符串:
     *      StringReader
     *      StringWriter
     */
    public class ByteArrayStreamDemo {
        public static void main(String[] args) throws IOException {
            // 写数据
            // ByteArrayOutputStream()
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
    
            // 写数据
            for (int x = 0; x < 10; x++) {
                baos.write(("hello" + x).getBytes());
            }
    
            // 释放资源
            // 通过查看源码我们知道这里什么都没做,所以根本需要close()
            // baos.close();
    
            // public byte[] toByteArray()
            byte[] bys = baos.toByteArray();
    
            // 读数据
            // ByteArrayInputStream(byte[] buf)
            ByteArrayInputStream bais = new ByteArrayInputStream(bys);
    
            int by = 0;
            while ((by = bais.read()) != -1) {
                System.out.print((char) by);
            }
    
            // bais.close();
        }
    }
    

    8.打印流

    Demo:

    import java.io.IOException;
    import java.io.PrintWriter;
    
    /*
     * 打印流
     * 字节流打印流   PrintStream
     * 字符打印流    PrintWriter
     * 
     * 打印流的特点:
     *      A:只有写数据的,没有读取数据。只能操作目的地,不能操作数据源。
     *      B:可以操作任意类型的数据。
     *      C:如果启动了自动刷新,能够自动刷新。
     *      D:该流是可以直接操作文本文件的。
     *          哪些流对象是可以直接操作文本文件的呢?
     *          FileInputStream
     *          FileOutputStream
     *          FileReader
     *          FileWriter
     *          PrintStream
     *          PrintWriter
     *          看API,查流对象的构造方法,如果同时有File类型和String类型的参数,一般来说就是可以直接操作文件的。
     * 
     *          流:
     *              基本流:就是能够直接读写文件的
     *              高级流:在基本流基础上提供了一些其他的功能
     */
    public class PrintWriterDemo {
        public static void main(String[] args) throws IOException {
            // 作为Writer的子类使用
            PrintWriter pw = new PrintWriter("pw.txt");
    
            pw.write("hello");
            pw.write("world");
            pw.write("java");
    
            pw.close();
        }
    }
    

    9.标准输入输出流

    9.1 “标准”输入流:InputStream

    Demo:

    import java.io.BufferedReader;
    import java.io.IOException;
    import java.io.InputStreamReader;
    
    /*
     * System.in 标准输入流。是从键盘获取数据的
     * 
     * 键盘录入数据:
     *      A:main方法的args接收参数。
     *          java HelloWorld hello world java
     *      B:Scanner(JDK5以后的)
     *          Scanner sc = new Scanner(System.in);
     *          String s = sc.nextLine();
     *          int x = sc.nextInt()
     *      C:通过字符缓冲流包装标准输入流实现
     *          BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
     */
    public class SystemInDemo {
        public static void main(String[] args) throws IOException {
            // //获取标准输入流
            // InputStream is = System.in;
            // //我要一次获取一行行不行呢?
            // //行。
            // //怎么实现呢?
            // //要想实现,首先你得知道一次读取一行数据的方法是哪个呢?
            // //readLine()
            // //而这个方法在哪个类中呢?
            // //BufferedReader
            // //所以,你这次应该创建BufferedReader的对象,但是底层还是的使用标准输入流
            // // BufferedReader br = new BufferedReader(is);
            // //按照我们的推想,现在应该可以了,但是却报错了
            // //原因是:字符缓冲流只能针对字符流操作,而你现在是字节流,所以不能是用?
            // //那么,我还就想使用了,请大家给我一个解决方案?
            // //把字节流转换为字符流,然后在通过字符缓冲流操作
            // InputStreamReader isr = new InputStreamReader(is);
            // BufferedReader br= new BufferedReader(isr);
            BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
    
            System.out.println("请输入一个字符串:");
            String line = br.readLine();
            System.out.println("你输入的字符串是:" + line);
    
            System.out.println("请输入一个整数:");
            // int i = Integer.parseInt(br.readLine());
            line = br.readLine();
            int i = Integer.parseInt(line);
            System.out.println("你输入的整数是:" + i);
        }
    }
    

    9.2 “标准”输出流:PrintStream

    Demo:

    import java.io.PrintStream;
    
    /*
     * 标准输入输出流
     * System类中的两个成员变量:
     *      public static final InputStream in “标准”输入流。
     *      public static final PrintStream out “标准”输出流。
     * 
     *      InputStream is = System.in;
     *      PrintStream ps = System.out;
     */
    public class SystemOutDemo {
        public static void main(String[] args) {
            // 有这里的讲解我们就知道了,这个输出语句其本质是IO流操作,把数据输出到控制台。
            System.out.println("helloworld");
    
            // 获取标准输出流对象
            PrintStream ps = System.out;
            ps.println("helloworld");
    
            ps.println();
            // ps.print();//这个方法不存在
    
            // System.out.println();
            // System.out.print();
        }
    }
    

    10.随机访问流:RandomAccessFile

    RandomAccessFile类不属于流,是Object类的子类。但它融合了InputStream和OutputStream的功能。支持对文件的随机访问读取和写入。

    构造方法

    public RandomAccessFile(String name,String mode):
    第一个参数是文件路径;
    第二个参数是操作文件的模式。
    模式有四种,我们最常用的一种叫"rw",这种方式表示我既可以写数据,也可以读取数据
    

    Demo:

    import java.io.IOException;
    import java.io.RandomAccessFile;
    
    public class RandomAccessFileDemo {
        public static void main(String[] args) throws IOException {
            // write();
            read();
        }
    
        private static void read() throws IOException {
            // 创建随机访问流对象
            RandomAccessFile raf = new RandomAccessFile("raf.txt", "rw");
    
            int i = raf.readInt();
            System.out.println(i);
            // 该文件指针可以通过 getFilePointer方法读取,并通过 seek 方法设置。
            System.out.println("当前文件的指针位置是:" + raf.getFilePointer());
    
            char ch = raf.readChar();
            System.out.println(ch);
            System.out.println("当前文件的指针位置是:" + raf.getFilePointer());
    
            String s = raf.readUTF();
            System.out.println(s);
            System.out.println("当前文件的指针位置是:" + raf.getFilePointer());
    
            // 我不想重头开始了,我就要读取a,怎么办呢?
            raf.seek(4);
            ch = raf.readChar();
            System.out.println(ch);
        }
    
        private static void write() throws IOException {
            // 创建随机访问流对象
            RandomAccessFile raf = new RandomAccessFile("raf.txt", "rw");
    
            // 怎么玩呢?
            raf.writeInt(100);
            raf.writeChar('a');
            raf.writeUTF("中国");
    
            raf.close();
        }
    }
    

    11.合并流:SequenceInputStream

    SequenceInputStream类可以将多个输入流串流在一起,合并为一个输入流,因此,该流也被称为合并流。

    构造方法

    SequenceInputStream(InputStream s1, InputStream s2)  ;
    SequenceInputStream(Enumeration<? extends InputStream> e);
    

    Demo1:

    import java.io.BufferedOutputStream;
    import java.io.FileInputStream;
    import java.io.FileOutputStream;
    import java.io.IOException;
    import java.io.InputStream;
    import java.io.SequenceInputStream;
    
    /*
     * 以前的操作:
     * a.txt -- b.txt
     * c.txt -- d.txt
     * 
     * 现在想要:
     * a.txt+b.txt -- c.txt
     */
    public class SequenceInputStreamDemo {
        public static void main(String[] args) throws IOException {
            // SequenceInputStream(InputStream s1, InputStream s2)
            // 需求:把ByteArrayStreamDemo.java和DataStreamDemo.java的内容复制到Copy.java中
            InputStream s1 = new FileInputStream("ByteArrayStreamDemo.java");
            InputStream s2 = new FileInputStream("DataStreamDemo.java");
            SequenceInputStream sis = new SequenceInputStream(s1, s2);
            BufferedOutputStream bos = new BufferedOutputStream(
                    new FileOutputStream("Copy.java"));
    
            // 如何写读写呢,其实很简单,你就按照以前怎么读写,现在还是怎么读写
            byte[] bys = new byte[1024];
            int len = 0;
            while ((len = sis.read(bys)) != -1) {
                bos.write(bys, 0, len);
            }
    
            bos.close();
            sis.close();
        }
    }
    

    Demo2:

    import java.io.BufferedOutputStream;
    import java.io.FileInputStream;
    import java.io.FileOutputStream;
    import java.io.IOException;
    import java.io.InputStream;
    import java.io.SequenceInputStream;
    import java.util.Enumeration;
    import java.util.Vector;
    
    /*
     * 以前的操作:
     * a.txt -- b.txt
     * c.txt -- d.txt
     * e.txt -- f.txt
     * 
     * 现在想要:
     * a.txt+b.txt+c.txt -- d.txt
     */
    public class SequenceInputStreamDemo2 {
        public static void main(String[] args) throws IOException {
            // 需求:把下面的三个文件的内容复制到Copy.java中
            // ByteArrayStreamDemo.java,CopyFileDemo.java,DataStreamDemo.java
    
            // SequenceInputStream(Enumeration e)
            // 通过简单的回顾我们知道了Enumeration是Vector中的一个方法的返回值类型。
            // Enumeration<E> elements()
            Vector<InputStream> v = new Vector<InputStream>();
            InputStream s1 = new FileInputStream("ByteArrayStreamDemo.java");
            InputStream s2 = new FileInputStream("CopyFileDemo.java");
            InputStream s3 = new FileInputStream("DataStreamDemo.java");
            v.add(s1);
            v.add(s2);
            v.add(s3);
            Enumeration<InputStream> en = v.elements();
            SequenceInputStream sis = new SequenceInputStream(en);
            BufferedOutputStream bos = new BufferedOutputStream(
                    new FileOutputStream("Copy.java"));
    
            // 如何写读写呢,其实很简单,你就按照以前怎么读写,现在还是怎么读写
            byte[] bys = new byte[1024];
            int len = 0;
            while ((len = sis.read(bys)) != -1) {
                bos.write(bys, 0, len);
            }
    
            bos.close();
            sis.close();
        }
    }
    

    12.序列化流

    序列化流:把对象按照流一样的方式存入文本文件或者在网络中传输。对象 -- 流数据(ObjectOutputStream)
    反序列化流:把文本文件中的流对象数据或者网络中的流对象数据还原成对象。流数据 -- 对象(ObjectInputStream)

    Demo:

    import java.io.FileInputStream;
    import java.io.FileOutputStream;
    import java.io.IOException;
    import java.io.ObjectInputStream;
    import java.io.ObjectOutputStream;
    
    public class ObjectStreamDemo {
        public static void main(String[] args) throws IOException,
                ClassNotFoundException {
            // 由于我们要对对象进行序列化,所以我们先自定义一个类
            // 序列化数据其实就是把对象写到文本文件
            write();
    
            read();
        }
    
        private static void read() throws IOException, ClassNotFoundException {
            // 创建反序列化对象
            ObjectInputStream ois = new ObjectInputStream(new FileInputStream(
                    "oos.txt"));
    
            // 还原对象
            Object obj = ois.readObject();
    
            // 释放资源
            ois.close();
    
            // 输出对象
            System.out.println(obj);
        }
    
        private static void write() throws IOException {
            // 创建序列化流对象
            ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(
                    "oos.txt"));
    
            // 创建对象
            Person p = new Person("林青霞", 27);
    
            // public final void writeObject(Object obj)
            oos.writeObject(p);
    
            // 释放资源
            oos.close();
        }
    }
    
    import java.io.Serializable;
    
    /*
     * 类通过实现 java.io.Serializable 接口以启用其序列化功能。未实现此接口的类将无法使其任何状态序列化或反序列化。
     * 该接口居然没有任何方法,类似于这种没有方法的接口被称为标记接口。
    
     * 注意:
     *      我一个类中可能有很多的成员变量,有些我不想进行序列化。请问该怎么办呢?
     *      使用transient关键字声明不需要序列化的成员变量
     */
    public class Person implements Serializable {
        private static final long serialVersionUID = -2071565876962058344L;
    
        private String name;
    
        // private int age;
    
        private transient int age;
    
        // int age;
    
        public Person() {
            super();
        }
    
        public Person(String name, int age) {
            super();
            this.name = name;
            this.age = age;
        }
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public int getAge() {
            return age;
        }
    
        public void setAge(int age) {
            this.age = age;
        }
    
        @Override
        public String toString() {
            return "Person [name=" + name + ", age=" + age + "]";
        }
    }
    

    13.Properties

    Hashtable的子类,说明是一个Map集合。
    属性集合类。是一个可以和IO流相结合使用的集合类。 Properties 可保存在流中或从流中加载。属性列表中每个键及其对应值都是一个字符串。

    Demo:

    import java.io.FileReader;
    import java.io.FileWriter;
    import java.io.IOException;
    import java.io.Reader;
    import java.io.Writer;
    import java.util.Properties;
    
    /*
     * 这里的集合必须是Properties集合:
     * public void load(Reader reader):把文件中的数据读取到集合中
     * public void store(Writer writer,String comments):把集合中的数据存储到文件
    */
    public class PropertiesDemo3 {
        public static void main(String[] args) throws IOException {
            myLoad();
    
            // myStore();
        }
    
        private static void myStore() throws IOException {
            // 创建集合对象
            Properties prop = new Properties();
    
            prop.setProperty("林青霞", "27");
            prop.setProperty("武鑫", "30");
            prop.setProperty("刘晓曲", "18");
    
            // public void store(Writer writer,String comments):把集合中的数据存储到文件
            Writer w = new FileWriter("name.txt");
            prop.store(w, "helloworld");
            w.close();
        }
    
        private static void myLoad() throws IOException {
            Properties prop = new Properties();
    
            // public void load(Reader reader):把文件中的数据读取到集合中
            // 注意:这个文件的数据必须是键值对形式
            Reader r = new FileReader("prop.txt");
            prop.load(r);
            r.close();
    
            System.out.println("prop:" + prop);
        }
    }
    

    14.NIO

    NIO其实就是新IO的意思。
    JDK4出现NIO。新IO和传统的IO有相同的目的,都是用于进行输入输出的,但新IO使用了不同的方式来处理输入输出,采用内存映射文件的方式,将文件或者文件的一段区域映射到内存中,就可以像访问内存一样的来访问文件了,这种方式效率比旧IO要高很多

    Java NIO提供了与标准IO不同的IO工作方式:
    Channels and Buffers(通道和缓冲区):标准的IO基于字节流和字符流进行操作的,而NIO是基于通道(Channel)和缓冲区(Buffer)进行操作,数据总是从通道读取到缓冲区中,或者从缓冲区写入到通道中。
    Asynchronous IO(异步IO):Java NIO可以让你异步的使用IO,例如:当线程从通道读取数据到缓冲区时,线程还是可以进行其他事情。当数据被写入到缓冲区时,线程可以继续处理它。从缓冲区写入通道也类似。
    Selectors(选择器):Java NIO引入了选择器的概念,选择器用于监听多个通道的事件(比如:连接打开,数据到达)。因此,单个的线程可以监听多个数据通道。

    Demo:

    import java.io.IOException;
    import java.nio.charset.Charset;
    import java.nio.file.Files;
    import java.nio.file.Paths;
    import java.util.ArrayList;
    
    /*
     * nio包在JDK4出现,提供了IO流的操作效率。但是目前还不是大范围的使用。
     * 有空的话了解下,有问题再问我。
     * 
     * JDK7的之后的nio:
     * Path:路径
     * Paths:有一个静态方法返回一个路径
     *      public static Path get(URI uri)
     * Files:提供了静态方法供我们使用
     *      public static long copy(Path source,OutputStream out):复制文件
     *      public static Path write(Path path,Iterable<? extends CharSequence> lines,Charset cs,OpenOption... options)
     */
    public class NIODemo {
        public static void main(String[] args) throws IOException {
            // public static long copy(Path source,OutputStream out)
            // Files.copy(Paths.get("ByteArrayStreamDemo.java"), new
            // FileOutputStream(
            // "Copy.java"));
    
            ArrayList<String> array = new ArrayList<String>();
            array.add("hello");
            array.add("world");
            array.add("java");
            Files.write(Paths.get("array.txt"), array, Charset.forName("GBK"));
        }
    }
    

    相关文章

      网友评论

        本文标题:JAVA基础系列(七) IO流

        本文链接:https://www.haomeiwen.com/subject/qqoyfxtx.html