• Java之IO流总结


      序言-

      第一次写博客,虽然工作快两年了,一直没想起来去分享什么,后来听同事说,没事喜欢去写写博客,我觉得,作为一个程序员,应该具备分享精神,所以觉得从基础开始,给刚接触Java的同学,一点点小小i的帮助,也算给自己温故而知新的机会。

      下面开始我们的Java学习之旅吧。

        IO流
    ·Java流式输入/输出原理
    ·Java流类的分类
    ·输入/输出流类
    ·常见的节点流和处理流
    ·文件流
    ·缓冲流
    ·转换流
    ·数据流
    ·Print流
    ·Object流

    ①Java流式输入/输出原理
        
            00101...-->
        文件 ------------------ 程序

            <-- ...00101
        文件 ------------------ 程序
        
               00101...-->
        网络连接 ------------------ 程序

            00101...-->
        程序 ------------------ 程序

             00101...-->  “Hello”
        文件 -----------============== 程序 (从0101等转换为字符,一层包一层)

    ②流的分类
        java.io包中
            ·按数据流的方向不同可以分为输入流、输出流
            ·按处理数据单位不同可以分为字节流、字符流
            ·按照功能不同可以分为节点流、处理流

        所有流类型位于包java.io内都分别继承以下四种抽象流类型

                字节流        字符流
        ------------------------------------------
        输入流          InputStream      Reader
        输出流          OutputSteam      Writer


            字节流(8bit)

                 |----FileInputStream(节点流)
                 |
                 |----PipedInputStream(节点流)     |---LineNumberInputStream(处理流)
                 |                          |       
                 |----FilterInputStream(处理流)-------|---DataInputStream(处理流)
                 |                     |
        InputStream -|----ByteArrayInputStream(节点流)    |---BufferedInputStream(处理流)
                 |                     |---PushbackInputStream(处理流)        
                 |----SequenceInputStream(处理流)
                 |
                 |----StringBufferInputStream(节点流)
                 |
                 |----ObjectInputStream(处理流)

    InputStream的基本方法
            //读取一个字节并以整数的形式返回(0-255)
            //如果返回-1已到输入流的末尾
        ·int read() throws IOExcetion

            //读取一系列字节并存储到一个数组buffer
            //返回实际读取的字节数,如果读取器前已到输入流的末尾返回-1
        ·int read(byre[] buffer) throws IOExcetion

            //读取length个字节
            //并存储到一个字节数组buffer,从length位置开始
            //返回实际读取的字节数,如果读取前已到输入流的末尾返回-1
            //buffer - 读入数据的缓冲区。
            //offset - 数组 buffer 中将写入数据的初始偏移量。
            //length - 要读取的最大字节数。
        ·int read(byte[] buffer, int offset, int length) throws IOExcetion

            //关闭流释放内存资源
        ·void close() throws IOExcetion

            //跳过n个字节不读,返回实际跳过的字节数
        ·long skip(long n) throws IOExcetion


            字节流(8bit)

                 |----FileOutputStream(节点流)
                 |
                 |----PipedOutputStream(节点流)    
                 |                          |       
                 |----FilterOutputStream(处理流)-----|---DataOutputStream(处理流)
                 |                     |
           OutputStream -|----ByteArrayOutputStream(节点流)  |---BufferedOutputStream(处理流)
                 |                     |---PrintStream(处理流)        
                 |
                 |----ObjectOutputStream(处理流)

    OutputStream的基本方法
            //向输出流中写入一个字节数据,该字节数据为参数b的低8位
        ·void write(int b) throws IOException
        
            //将一个字节类型的数组中的数据写入输出流
        ·void write(byre[] b) throws IOException

            //将一个字节类型的数组中的从指定位置(off)开始的len个字节写入到输出流
        ·void write(byre[] b, int off, int len) throws IOException

            //关闭流释放内存资源
        ·void close() throws IOException

            //将输出流中缓冲的数据全部写出到目的地
        ·void flush() throws IOException

        注意:先flush,在close



            字符流(16bit)

                 |----BufferedReader(节点流)----LineNumberReader(处理流)
                 |
                 |----CharArrayReader(处理流)    
                 |                              
                 |----InputStreamReader(处理流)-----FileReader(节点流)
                 |                    
                 Reader -|----FilterReader(处理流)----PushbackReader(处理流)
                 |                         
                 |
                 |----PipedReader(节点流)
                 |
                 |----StringReader(节点流)


    Reader的基本方法
            //读取一个字符并以整数的形式返回(0-255)
            //如果返回-1已到输入流的末尾
        ·int read() throws IOExcetion

            //读取一系列字符并存储到一个数组buffer
            //返回实际读取的字符数,如果读取器前已到输入流的末尾返回-1
        ·int read(byre[] buffer) throws IOExcetion

            //读取length个字符
            //并存储到一个字符数组buffer,从length位置开始
            //返回实际读取的字符数,如果读取前已到输入流的末尾返回-1
            //buffer - 读入数据的缓冲区。
            //offset - 数组 buffer 中将写入数据的初始偏移量。
            //length - 要读取的最大字符数。
        ·int read(byte[] buffer, int offset, int length) throws IOExcetion

            //关闭流释放内存资源
        ·void close() throws IOExcetion

            //跳过n个字符不读,返回实际跳过的字节数
        ·long skip(long n) throws IOExcetion


            字符流(16bit)

                 |----BufferedWriter(处理流)
                 |
                 |----CharArrayWriter(节点流)    
                 |                              
                 |----OutputStreamReader(处理流)-----FileWriter(节点流)
                 |                    
                 Writer -|----FilterWriter(处理流)
                 |                         
                 |
                 |----PipedWriter(节点流)
                 |
                 |----StringWriter(节点流)

    Writer的基本方法
            //向输出流中写入一个字符数据,该字节数据为参数b的低8位
        ·void write(int b) throws IOException
        
            //将一个字符类型的数组中的数据写入输出流
        ·void write(char[] cbuf) throws IOException

            //将一个字符类型的数组中的从指定位置(off)开始的len个字符写入到输出流
        ·void write(char[] cbuf, int off, int len) throws IOException

            //将一个字符串中的字符写入到输出流
        ·void write(String string) throws IOException

            //将一个字符串从offset开始的length个字符写入到输出流
        ·coid write(String string, int offset, int length) throws IOException

            //关闭流释放内存资源
        ·void close() throws IOException

            //将输出流中缓冲的数据全部写出到目的地
        ·void flush() throws IOException


    ③输入/输出流
        1、输出流跟输入流
        一切以程序为中心
            ·从文件到程序为输入流
            ·从程序到文件为输出流
        2、字节流和字符流
            ·字节(8bit)
            ·字符(16bit)
            ·一个字符等于2个字节
        3、节点流和处理流
            ·处理流是包在节点流的一层“管道”

    ④常见的节点流和处理流
        ·节点流为可以从一个特定的数据源(节点)读写数据(如:文件,内存)
                节点流
            数据源 -------------------- 程序
            
        ·处理流是连接在已存在的流(节点流或处理流)之上,通过对数据的处理为程序提供更为强大的读写功能

                -->
            数据源 -------======== 程序
                
                <--
            程序 ============-------- 数据源

    ⑤节点流----文件流
        字符流            字节流
    ---------------------------------------
    输入流    FileReader    FileInputStream
    输出流    FileWriter    FileOutputStream

    访问文件
        从文件到程序,输入流
        ·使用FileInputStream

            public class Login{
        public static void main(String[] args){
            int b = 0;
            FileInputStream fis = null;
            try {
                //连接管道
                fis = new FileInputStream("G://Java2014Project//JavaDemo1//src//Demo01/Login.java");
            } catch (FileNotFoundException e) {
                System.out.println("error");
            }
            //读取数据
                try {
                    while((b = fis.read())!= -1){
                        System.out.print((char)(b));
                        
                    }
                    fis.close();
                } catch (IOException e) {
                    System.out.println("error");
                }
            
        }
    }
        显示结果
        package Demo01;

    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.IOException;

    public class Login{
        public static void main(String[] args){
            int b = 0;
            FileInputStream fis = null;
            try {
                //????????
                fis = new FileInputStream("G://Java2014Project//JavaDemo1//src//Demo01/Login.java");
            } catch (FileNotFoundException e) {
                System.out.println("error");
            }
            //????????
                try {
                    int num = 0;
                    while((b = fis.read())!= -1){
                        System.out.print((char)(b));
                    }
                    fis.close();
                } catch (IOException e) {
                    System.out.println("error");
                }
            
        }
    }

        发现出现?????,问题是汉字是一个字符(两个字节),用的是字节流,一个字节一个字节读取,
    所有中文翻译不出来,改为Reader可以
        
        从程序到文件
        ·使用FileOutputStream
        package Demo01;

    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.io.IOException;

    public class Login{
        public static void main(String[] args){
            int b = 0;
            FileInputStream fis = null;
            FileOutputStream fot = null;
            try {
                //连接管道
                fis = new FileInputStream("G://Java2014Project//JavaDemo1//src//Demo01/Login.java");
                fot = new FileOutputStream("C://Users//woshishabi//Desktop//test.txt");
            } catch (FileNotFoundException e) {
                System.out.println("error");
            }
            //读取数据
                try {
                    while((b = fis.read())!= -1){
                        fot.write(b);
                    }
                    fis.close();
                    fot.close();
                } catch (IOException e) {
                    System.out.println("error");
                }
            
        }
    }

        桌面上多了一个Test.txt文本,


        ·使用FileReader:字符输入流,从文件到程序
            ......
        ·使用FileWriter:字符输出流,从程序到文件
            ......

    ⑥处理流----缓冲流
        ·缓冲流要“套接”在相应的节点流之上,对读写的数据提供了缓冲的功能,提高了读写的效率,
    同时增加了一些新的方法
        ·J2SDK提供了四种缓存流,其常用的构造方法为:
                //字符输入缓冲流
            BufferedReader(Reader in)
            BufferedReader(Reader in, int sz) //sz为自定义缓存区的大小    
                
                //字符输出缓冲流
            BufferedWriter(Writer out)
            BufferedWriter(Writer out,int sz)
            
                //字节输入缓冲流
            BufferedInputStream(InputStream in)
            BufferedInputStream(InputStream in, int size)
        
                //字节输出缓冲流
            BufferedOutputStream(OutputStream out)
            BufferedOutputStream(OutputStream out, int size)

        ·缓冲输入流支持其父类的mark和reset方法    mark:直接从多少个字符开始读取  reset:回到刚才标记的点
        ·bufferedReader提供了readline方法用于读取一行字符串(以 或 分隔)
        ·bufferedWriter提供了newLine用于写入一个行分隔符
        ·对于输出的缓冲流,写出的数据会现在内存中缓存,使用flush方法将会使内存中的数据立刻写出

    bufferedReader和bufferedWriter使用

    package Demo01;

    import java.io.BufferedInputStream;
    import java.io.BufferedReader;
    import java.io.BufferedWriter;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.io.FileReader;
    import java.io.FileWriter;
    import java.io.IOException;

    public class Login{
        public static void main(String[] args){
            try {
                //从文件写入数据
                BufferedReader br  = new BufferedReader(new FileReader("G://Java2014Project//JavaDemo1//src//Demo01//Login.java"));
                BufferedWriter bw = new BufferedWriter(new FileWriter("C://Users//woshishabi//Desktop//Test.txt"));
                //从文件读取数据
                String s = null;
                while((s = br.readLine()) != null){
                    System.out.println(s);
                    bw.write(s);
                    bw.newLine();
                }
                bw.flush();
                br.close();
                bw.close();
            }
                catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        
        }
    }


    ⑦处理流----转换流
        ·InputStreamReader和OutputStreamWriter用与字节数据到字符数据之间的转换
        ·InputStreamReader需要和InputStream“套接”
        ·OutputStream需要和OutputStream“套接”
        ·转换流在构造时可以指定其编码集合,例如
            InputStream isr = new InputStreamReader(System.in, "ISO8859_1");

    范例:OutputStreamWriter

        package Demo01;

    import java.io.BufferedInputStream;
    import java.io.BufferedReader;
    import java.io.BufferedWriter;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.io.FileReader;
    import java.io.FileWriter;
    import java.io.IOException;
    import java.io.OutputStreamWriter;

    public class Login{
        public static void main(String[] args){
            try {
                //将输出字节流转换为字符流
                OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("C://Users//woshishabi//Desktop//Test.txt"));
                //向文件中输入字符串
                osw.write("sdfjaksldjflskdjflaskdjfsakjdf");
                //读取字符编码
                System.out.println(osw.getEncoding());
                osw.close();
                //在文本后面添加字符串,不覆盖,则true,否则写false, 后面可以自定义编码
                osw = new OutputStreamWriter(new FileOutputStream("C://Users//woshishabi//Desktop//Test.txt", true), "UTF-8");
                osw.write("qwqeqweqweqeq");
                System.out.println(osw.getEncoding());
                osw.close();
            }
                catch (IOException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
                
            }
        }

    范例:InputStreamReader

        package Demo01;

    import java.io.BufferedInputStream;
    import java.io.BufferedReader;
    import java.io.BufferedWriter;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.io.FileReader;
    import java.io.FileWriter;
    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.io.OutputStreamWriter;

    public class Login{
        public static void main(String[] args){
            //从键盘读取数据
            InputStreamReader isr = new InputStreamReader(System.in);
            //字符输入缓冲流
            BufferedReader br = new BufferedReader(isr);
            String s = null;
            try {
                //读取一行字符
                s = br.readLine();
                while(s != null){
                if(s.equalsIgnoreCase("exit")){     //将一个字符串与另一个字符串比较,不考虑大小写
                    break;    
                }
                //将小写转换为大写
                    System.out.println(s.toUpperCase());
                    s = br.readLine();
                }
                br.close();
                
                }
             catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            
        }
    }


    ⑧数据流----DataInputStream...
        ·DataInputStream 和 DataOutputStream 分别继承了InputStream 和 OutputStream,属于处理流,
    需要分别套接在InputStream 和 OutputStream类型的节点流觞
        ·DataInputStream 和 DataOutputStream 提供了可以存取与机器无关的Java原始数据类型(int、double等)
    的方法
        ·DataInputStream 和 DataOutputStream 的构造方法为:
            ·DataInputStream(InputStream in)
            ·DataOutputStream(OutputStream out)
        
    范例:使用DataInputStream 和 DataOutputStream
    package Demo01;

    import java.io.BufferedInputStream;
    import java.io.BufferedReader;
    import java.io.BufferedWriter;
    import java.io.ByteArrayInputStream;
    import java.io.ByteArrayOutputStream;
    import java.io.DataInputStream;
    import java.io.DataOutputStream;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.io.FileReader;
    import java.io.FileWriter;
    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.io.OutputStreamWriter;

    public class Login{
        public static void main(String[] args){
            //一个字节数组输出流
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            //在字节数组输出流上套接一个数据流,能传递基本数据类型
            DataOutputStream dos = new DataOutputStream(baos);
            try {
                //向字节数组写入一个随机数
                dos.writeDouble(Math.random());
                dos.writeBoolean(true);
                //创建一个字节数组输入流
                ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());
                //输出字节数组中有多少个字节
                System.out.println(bais.available());
                //将处理流包在字节数组输入流上,输出基本数据
                DataInputStream dis = new DataInputStream(bais);
                //输出double,先输入的先输出
                System.out.println(dis.readDouble());
                //输出boolean
                System.out.println(dis.readBoolean());
                dos.close();
                dis.close();
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }
    }

        
    ⑨print流    PrintStream主要操作byte流,而PrintWriter用来操作字符流
        ·PrintWriter 和 PrintStream都属于输出流,分别针对与字符和字节
        ·PrintWriter 和 PrintStream提供了重载的print
        ·Println方法用于多种数据类型的输出
        ·PrintWriter 和 PrintStream的输出操作不会抛出异常,用户通过检测错误状态获取错误信息
        ·PrintWriter 和 PrintStream有自动flush功能
    构造方法:
        ·PrintWriter(Writer out)
        ·PrintWriter(Writer out,boolean autoFlush)
        ·PrintWriter(OutputStream out, boolean autoFlush)
        ·PrintStream(OutputStream out)
        ·PrintStream(OutputStream out, boolean autoFlush)

    范例:PrintStream
    package Demo01;

    import java.io.BufferedInputStream;
    import java.io.BufferedReader;
    import java.io.BufferedWriter;
    import java.io.ByteArrayInputStream;
    import java.io.ByteArrayOutputStream;
    import java.io.DataInputStream;
    import java.io.DataOutputStream;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.io.FileReader;
    import java.io.FileWriter;
    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.io.OutputStreamWriter;
    import java.io.PrintStream;

    public class Login{
        public static void main(String[] args){
            PrintStream ps = null;
            try {
                //文件字节输出流
                FileOutputStream fos = new FileOutputStream("C://Users//woshishabi//Desktop//Test.txt");
                //输出流
                ps = new PrintStream(fos);
                if(ps != null){
                    //使system.out指向了文件Test.txt
                    System.setOut(ps);
                }
                for (int i = 0; i < 100; i++) {
                    System.out.print(i + ",");
                }

            } catch (FileNotFoundException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }
    }


    范例:PrintWriter

        package Demo01;

    import java.io.BufferedInputStream;
    import java.io.BufferedReader;
    import java.io.BufferedWriter;
    import java.io.ByteArrayInputStream;
    import java.io.ByteArrayOutputStream;
    import java.io.DataInputStream;
    import java.io.DataOutputStream;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.io.FileReader;
    import java.io.FileWriter;
    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.io.OutputStreamWriter;
    import java.io.PrintStream;
    import java.io.PrintWriter;
    import java.util.Date;

    public class Login{
        public static void main(String[] args){
            String s = null;
            //从键盘输入,字节转字符的转换流上套接缓冲流
            BufferedReader bis = new BufferedReader(new InputStreamReader(System.in));
            try {
                //输出流,输出到文件,不覆盖
                FileWriter fw = new FileWriter("C://Users//woshishabi//Desktop//Test.txt",true);
                //输出流,套接在文件输出流上
                PrintWriter pw = new PrintWriter(fw);
                //接收键盘输入的一行数据,
                while((s = bis.readLine()) != null){
                    //判断是否为exit,是则结束while
                    if(s.equalsIgnoreCase("exit"))break;
                    //小写转大写,在dos中输出
                    System.out.println(s.toUpperCase());
                    //在目标文件里面输出数据
                    pw.println("------");
                    pw.println(s.toLowerCase());
                    pw.flush();
                }
                //在目标文件中输出日期
                pw.println("-----"+ new Date() + "------");
                pw.flush();
                pw.close();
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }
    }


    ⑩Object流
        ·直接将Object写入或读出
            ·transient关键字(透明的):修饰属性, transient int k  = 15;   输出结果k = 0;在序列化的时候不考虑,不写入硬盘
            ·serializable接口:(标记性接口,里面什么都没有)
            ·extemalizable接口:
        注意:如果需要使用对象流,必须实现serializable接口

    范例:
        package Demo01;

    import java.io.BufferedInputStream;
    import java.io.BufferedReader;
    import java.io.BufferedWriter;
    import java.io.ByteArrayInputStream;
    import java.io.ByteArrayOutputStream;
    import java.io.DataInputStream;
    import java.io.DataOutputStream;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.io.FileReader;
    import java.io.FileWriter;
    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.io.ObjectInputStream;
    import java.io.ObjectOutputStream;
    import java.io.OutputStreamWriter;
    import java.io.PrintStream;
    import java.io.PrintWriter;
    import java.io.Serializable;
    import java.util.Date;

    public class Login{
        public static void main(String[] args){
            T t = new T();
            t.k = 40;
            try {
                //文件输出流,
                FileOutputStream fos = new FileOutputStream("C://Users//woshishabi//Desktop//Test.txt");
                //在文件输出流上套接一个对象输出流,从程序传递一个对象到目标文件中
                ObjectOutputStream  oos = new ObjectOutputStream(fos);
                //写入对象
                oos.writeObject(t);
                //文件输入流,锁定目标文件
                FileInputStream fis = new FileInputStream("C://Users//woshishabi//Desktop//Test.txt");
                //在文件输入流上套接一个对象输入流,从目标文件中取出对象的通道
                ObjectInputStream ois = new ObjectInputStream(fis);

                //取出对象
                T t1 = (T) ois.readObject();
                //打印到屏幕上
                System.out.println(t1.i + " " + t1.j + " " + t1.k + " " + t1.z);
            } catch (ClassNotFoundException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }

        }
    }

    class T implements Serializable{
        int i = 1;
        int j = 2;
        double z = 9.0;
        int k = 14;

    }




    总结:
        ·InputStream/OutputStream

        ·Reader/Writer

        ·FileInputStream/FileoutputStream

        ·FileReader/FileWriter

        ·BufferedInputStream/BufferedOutoutStream

        ·BufferedTeader/BufferedWriter

        ·ByteArrayInputStream/ByteArrayOutputStream

        ·InputStreamReader/OutputStreamWriter

        ·DataInputStream/DataOutputStream

        ·PrintStream/PrintWriter

        ·ObjectInputStream/ObjectOutputStream

  • 相关阅读:
    google.guava 实现 限流
    基于 Redisson 的限流 小 demo
    TX-LCN分布式事务-- TCC事务模式(消费者模块)
    TX-LCN分布式事务-- TCC事务模式(生产者模块)
    TX-LCN分布式事务-- LCN事务模式(消费者模块)
    TX-LCN分布式事务-- LCN事务模式(生产者模块)
    TX-LCN分布式事务-- LCN事务模式(eureka模块)
    TX-LCN分布式事务-- LCN事务模式(tm模块)
    TX-LCN分布式事务--学习地址和原理介绍
    LINQ to SQL系列三 使用DeferredLoadingEnabled,DataLoadOption指定加载选项
  • 原文地址:https://www.cnblogs.com/whgk/p/5326568.html
Copyright © 2020-2023  润新知