java——io流

来源:互联网 发布:十金数据黄金外汇直播 编辑:程序博客网 时间:2024/05/16 09:54


        看到了一篇写的很好的关于io流的总结,并进行了自己的一些补充。


        IO流主要用于硬盘、内存、键盘等处理设备或网络上的数据操作,根据处理数据的数据类型的不同可以分为:字节流(抽象基类为InPutStream和OutPutStream)和字符流(抽象基类为Reader和Writer)。根据流向不同,可以分为:输入流和输出流。  其中主要结构可以用下图来表示:                          

                                   

      

 字符流和字节流的主要区别:

       1.字节流读取的时候,读到一个字节就返回一个字节;  字符流使用了字节流读到一个或多个字节(中文对应的字节数是两个,在UTF-8码表中是3个字节)时。先去查指定的编码表,将查到的字符返回。

       2.字节流可以处理所有类型数据,如:图片,MP3,AVI视频文件,而字符流只能处理字符数据。只要是处理纯文本数据,就要优先考虑使用字符流,除此之外都用字节流。

     

 IO流主要可以分为节点流和处理流两大类。


一、节点流类型

       该类型可以从或者向一个特定的地点或者节点读写数据。主要类型如下:

类型字符流字节流File(文件)
FileReader
FileWriter
InputStreamReader
OutputStreamWriter
FileInputStream
FileOutputSream
Memory Array
CharArrayReader
CharArrayWriter
ByteArrayInputStream
ByteArrayOutputSream
Memory String
StringReader
StringWriter
   -Pipe(管道)
PipedReader
PipedWriter
PipedInputSream
PipedOutputSream

      ——字节流FileInputStream和FileOutputStream读写时,只能在字节级别读写,我们只能从其对象中读入字节和字节数组。  
    无论是二进制数据还是字符数据(文本数据)都可以用文件输出流 java.io.FileOutputStream 以字节流的方式保存到所指定文件。不过,如果写字符数据,FileWriter 更方便一些。FileWriter 除了提供了文件写入功能之外,还内置了字符编码功能。而FileInputStream只是纯粹的将字节写入文件。
      字符流FileReader和FileWriter使用了字节流读到一个或多个字节(中文对应的字节数是两个,在UTF-8码表中是3个字节)时。先去系统默认的编码表,将查到的字符返回。我们只能从其对象中逐个读入或写出字符。
     另外InputStreamReader和OutputStreamWriter是FileReader和FileWriter的超类,从下面描述可看出,其可以替代FileReader和FileWriter的功能。
       所以,若想更方便的处理流,必须在节点流的基础上借助处理流!

二、处理流类型
        该类型是对一个已存在的流的连接和封装(至少要有一个节点流),通过所封装的流的功能调用实现数据读写,处理流的构造方法总是要带一个其他流对象作为参数,一个流对象进过其他流的多次包装,叫做流的链接。主要可以分为以下几种:
 
 1、缓冲流(BufferedInPutStream/BufferedOutPutStream和BufferedWriter/BufferedReader)他可以提高对流的操作效率。
               
              写入缓冲区对象:                 
[java] view plain copy
  1. BufferedWriter bufw=new BufferedWriter(new FileWriter("buf.txt"));  
  
              读取缓冲区对象:                
[java] view plain copy
  1. BufferedReader bufr=new BufferedReader(new FileReader("buf.txt"));  
         
       该类型的流有一个特有的方法:readLine():一次读一行,到行标记时,将行标记之前的字符数据作为字符串String类型返回;当读到末尾时,返回null。其原理还是与缓冲区关联的流对象的read方法,只不过每一次读取到一个字符,先不进行具体操作,先进行临时储存,当读取到回车标记时,将临时容器中储存的数据一次性返回。
                   
         
2、转换流(InputStreamReader/OutputStreamWriter)
      该类型是字节流和字符流之间的桥梁。
      该类型是FileReader和FileWriter的超类,从下面描述可看出,其可以替代FileReader和FileWriter的功能。
      并且该流对象可以将Unicode字符流指定的或系统默认的字节流进行转换。
      构造函数主要有:    
[java] view plain copy
  1. InputStreamReader(InputStream);        //通过构造函数初始化,使用的是本系统默认的编码表GBK。  
  2.  InputStreamReader(InputStream,String charSet);   //通过该构造函数初始化,可以指定编码表。  
  3.  OutputStreamWriter(OutputStream);      //通过该构造函数初始化,使用的是本系统默认的编码表GBK。  
  4.  OutputStreamwriter(OutputStream,String charSet);   //通过该构造函数初始化,可以指定编码表。  
       注意:在使用FileReader操作文本数据时,该对象使用的是系统默认的编码表,(即FileReader的对象将包含字节(用某种字符编码方式表示的字符)的输入流转换为产生Unicode码元的读入器。)且FileReader只能使用默认编码表,不能指定编码表
       FileReader fr=new FileReader(“a.txt”);      与     InputStreamReader isr=new InputStreamReader(new FileInputStream("a.txt"));   的意义相同。          如果要使用指定编码表时,必须使用转换流,即如果a.txt中的文件中的字符数据是通过utf-8的形式编码,那么在读取时,就必须指定编码表,那么转换流时必须的。即:
                   InputStreamReader isr=new InputStreamReader(new FileInputStream("a.txt"),utf-8);
         
 3、数据流(DataInputStream/DataOutputStream)

          该数据流可以方便地对一些基本类型数据进行直接的存储和读取,不需要再进一步进行转换,通常只要操作基本数据类型的数据,就需要通过DataStream进行包装。
          构造方法:        
[java] view plain copy
  1. DataInputStreamReader(InputStream);  
  2.  DataInputStreamWriter(OutputStream);  
          方法举例:          
[java] view plain copy
  1. int readInt();//一次读取四个字节,并将其转成int值  
  2. writeInt(int);//一次写入四个字节,注意和write(int)不同,write(int)只将该整数的最低一个8位写入,剩余三个8为丢失  
  3. short readShort();  
  4. writeShort(short);  
  5. String readUTF();//按照utf-8修改版读取字符,注意,它只能读writeUTF()写入的字符数据。  
  6.  writeUTF(String);//按照utf-8修改版将字符数据进行存储,只能通过readUTF读取。  
         注意:在使用数据流读/存数据的时候,需要有一定的顺序,即某个类型的数据先写入就必须先读出,服从先进先出的原则。
         在Java中所有的值都是按照高位在前的模式MSB写出(最左边的字节存储最高位),不管处理器的存储方式。
 
四、打印流(PrintStream/PrintWriter)

       PrintStream是一个字节打印流,System.out对应的类型就是PrintStream,它的构造函数可以接受三种数据类型的值:1.字符串路径  2.File对象 3.OutputStream。
       PrintWriter是一个字符打印流,它的构造函数可以接受四种类型的值:1.字符串路径。2.File对象 3.OutputStream  4.Writer  对于1、2类型的数据,可以指定编码表,也就是字符集,对于3、4类型的数据,可以指定自动刷新,当该自动刷新为True时,只有3个方法可以用println,printf,format。
当自动刷新设置为true时,只要println被调用,缓冲器中的所有字符都会被发送到目的地(打印写出器总是带缓冲区的,且默认自动刷新机制为false)
       所以,因OutputStream 和.Writer 只能逐个字节或字符写入,,write(int)只将该整数的最低一个8位写入,剩余三个8为丢失!所以想方便的一次性传输多个字节或字符,要依靠PrintStream和PrintWriter。 

五、对象流(ObjectInputStream/ObjectOutputStream)

        该类型的流可以把类的对象作为一个整体进行存取,主要方法有:
        Object readObject();该方法抛出异常:ClassNotFountException。
        void writeObject(Object):被写入的对象必须实现一个接口:Serializable,否则就会抛出:NotSerializableException
        在读写端都要有该类的相同描述,并且都实现一个接口:Serializable。ObjectOutputStream 浏览对象的所有域并存储他们的内容。
        注意:当一个对象被多个对象共享,作为他们各自的一部分时,(例如每个Manager对象都包含一个表示秘书的Employee的引用,且两个经理共用一个秘书时)保存这样的对象网络是一种挑战,在这里不能去保存和恢复秘书对象的内存地址,因为当对象被重新加载时,它可能占据的是与原来完全不同的内存地址。实际上的机制是每个对象都用一个序列号保存,这就是被称为对象序列化的原因
        方法如下:
        ObjectOutputStream时:
       1.对每一个遇到的对象保存其对象数据到流中;
       2.如果某个对象之前已经被保存过,那么只写出“与之前保存过的序列号为x的对象相同”。
        ObjectInputStream时:
       1.对流中的对象,在第一次遇到其序列号时,构建它,并使用流中的数据来初始化它,然后记录这个序列号和构建的对象之间的关联;
       2.当遇到“与之前保存过的序列号为x的对象相同”标记时,获取与这个序列号相关联的对象的引用。
    
       序列化机制提供了一种深拷贝对象的途径:前提是该类实现了接口:Serializable。做法很简单:直接将对象序列化到输出流中,然后将其读回。这样产生的新对象是对现有对象的一个深拷贝。在此过程中,我们不必将对象写出到文件中,因为可以用ByteArrayOutputStream将数据保存在字节数组中。
       

转自:http://blog.csdn.net/zzp_403184692/article/details/8057693
1 0
原创粉丝点击