1.什么是IO:

Java中I/O操作主要是指使用Java进行输入,输出操作. Java所有的I/O机制都是基于数据流进行输入输出,这些数据流表示了字符或者字节数据的流动序列。Java的I/O流提供了读写数据的标准方法。任何Java中表示数据源的对象都会提供以数据流的方式读写它的数据的方法。Java.io是大多数面向数据流的输入/输出类的主要软件包。此外,Java也对块传输提供支持,在核心库 java.nio中采用的便是块IO。流IO的好处是简单易用,缺点是效率较低。块IO效率很高,但编程比较复杂。


 Java IO模型  :

      Java的IO模型设计非常优秀,它使用Decorator模式,按功能划分Stream,您可以动态装配这些Stream,以便获得您需要的功能。例如,您需要一个具有缓冲的文件输入流,则应当组合使用FileInputStream和BufferedInputStream。 


2.数据流的基本概念:


 数据流是一串连续不断的数据的集合,就象水管里的水流,在水管的一端一点一点地供水,而在水管的另一端看到的是一股连续不断的水流。数据写入程序可以是一段、一段地向数据流管道中写入数据,这些数据段会按先后顺序形成一个长的数据流。对数据读取程序来说,看不到数据流在写入时的分段情况,每次可以读取其中的任意长度的数据,但只能先读取前面的数据后,再读取后面的数据。不管写入时是将数据分多次写入,还是作为一个整体一次写入,读取时的效果都是完全一样的。 
     在电脑上的数据有三种存储方式,一种是外存,一种是内存,一种是缓存。比如电脑上的硬盘,磁盘,U盘等都是外存,在电脑上有内存条,缓存是在CPU里面的。外存的存储量最大,其次是内存,最后是缓存,但是外存的数据的读取最慢,其次是内存,缓存最快。这里总结从外存读取数据到内存以及将数据从内存写到外存中。对于内存和外存的理解,我们可以简单的理解为容器,即外存是一个容器,内存又是另外一个容器。那又怎样把放在外存这个容器内的数据读取到内存这个容器以及怎么把内存这个容器里的数据存到外存中呢?

在Java类库中,IO部分的内容是很庞大的,因为它涉及的领域很广泛。标准输入输出,文件的操作,网络上的数据流,字符串流,对象流,zip文件流等等,java中将输入输出抽象称为流,就好像水管,将两个容器连接起来。将数据冲外存中读取到内存中的称为输入流,将数据从内存写入外存中的称为输出流。流是一个很形象的概念,当程序需要读取数据的时候,就会开启一个通向数据源的流,这个数据源可以是文件,内存,或是网络连接。类似的,当程序需要写入数据的时候,就会开启一个通向目的地的流。总结的基本概念如下:


1) 数据流:

    一组有序,有起点和终点的字节的数据序列。包括输入流和输出流。

  

Java io流路径 java中的io流_Java io流路径

2) 输入流(Input  Stream):

      程序从输入流读取数据源。数据源包括外界(键盘、文件、网络…),即是将数据源读入到程序的通信通道



     

Java io流路径 java中的io流_Java io流路径_02



 3) 输出流:


    程序向输出流写入数据。将程序中的数据输出到外界(显示器、打印机、文件、网络…)的通信通道。

     

Java io流路径 java中的io流_Java_03

    

采用数据流的目的就是使得输出输入独立于设备。

Input  Stream不关心数据源来自何种设备(键盘,文件,网络)
Output  Stream不关心数据的目的是何种设备(键盘,文件,网络)

4 数据流分类:


流序列中的数据既可以是未经加工的原始二进制数据,也可以是经一定编码处理后符合某种格式规定的特定数据。因此Java中的流分为两种:
 1)  字节流:数据流中最小的数据单元是字节
 2)  字符流:数据流中最小的数据单元是字符, Java中的字符是Unicode编码,一个字符占用两个字节。




3. 标准I/O:


Java程序可通过命令行参数与外界进行简短的信息交换,同时,也规定了与标准输入、输出设备,如键盘、显示器进行信息交换的方式。而通过文件可以与外界进行任意数据形式的信息交换。

1. 命令行参数


1. public class TestArgs {  
2. public static void main(String[] args) {  
3. for (int i = 0; i < args.length; i++) {  
4. "args[" + i + "] is <" + args[i] + ">");  
5.         }  
6.     }  
7. }

运行命令:java Java C VB

运行结果:

args[0] is <Java>

args[1] is <C>

args[2] is <VB>



2. 标准输入,输出数据流

java系统自带的标准数据流:java.lang.System:

  1. java.lang.System   
  2. public final class System  extends Object{   
  3. static  PrintStream  err;//标准错误流(输出)  
  4. static  InputStream  in;//标准输入(键盘输入流)  
  5. static  PrintStream  out;//标准输出流(显示器输出流)  
  6. }  

注意:

(1)System类不能创建对象,只能直接使用它的三个静态成员。
(2)每当main方法被执行时,就自动生成上述三个对象。

1) 标准输出流 System.out

System.out向标准输出设备输出数据,其数据类型为PrintStream。方法:
      Void print(参数)
      Void println(参数)

2)标准输入流 System.in


System.in读取标准输入设备数据(从标准输入获取数据,一般是键盘),其数 据类型为InputStream。方法:

        int read()  //返回ASCII码。若,返回值=-1,说明没有读取到任何字节读取工作结束。

         int read(byte[] b)//读入多个字节到缓冲区b中返回值是读入的字节数

例如:

System.in读取标准输入设备数据(从标准输入获取数据,一般是键盘),其数 据类型为InputStream。方法:

1. import java.io.*;  
2. public class StandardInputOutput {  
3. public static void main(String args[]) {  
4. int b;  
5. try {  
6. "please Input:");  
7. while ((b = System.in.read()) != -1) {  
8. char) b);  
9.             }  
10. catch (IOException e) {  
11.             System.out.println(e.toString());  
12.         }  
13.     }  
14. }

等待键盘输入,键盘输入什么,就打印出什么:

Java io流路径 java中的io流_Java io流路径_04


3)标准错误流

   System.err输出标准错误,其数据类型为PrintStream。可查阅API获得详细说明。

    标准输出通过System.out调用println方法输出参数并换行,而print方法输出参数但不换行。println或print方法都通 过重载实现了输出基本数据类型的多个方法,包括输出参数类型为boolean、char、int、long、float和double。同时,也重载实现 了输出参数类型为char[]、String和Object的方法。其中,print(Object)和println(Object)方法在运行时将调 用参数Object的toString方法。

1. import java.io.BufferedReader;  
2. import java.io.IOException;  
3. import java.io.InputStreamReader;  
4.   
5. public class StandardInputOutput {  
6. public static void main(String args[]) {  
7.         String s;  
8. // 创建缓冲区阅读器从键盘逐行读入数据  
9. new InputStreamReader(System.in);  
10. new BufferedReader(ir);  
11. "Unix系统: ctrl-d 或 ctrl-c 退出"  
12. "\nWindows系统: ctrl-z 退出");  
13. try {  
14. // 读一行数据,并标准输出至显示器  
15.             s = in.readLine();  
16. // readLine()方法运行时若发生I/O错误,将抛出IOException异常  
17. while (s != null) {  
18. "Read: " + s);  
19.                 s = in.readLine();  
20.             }  
21. // 关闭缓冲阅读器  
22.             in.close();  
23. catch (IOException e) { // Catch any IO exceptions.  
24.             e.printStackTrace();  
25.         }  
26.     }  
27. }


4.java.IO层次体系结构




在整个Java.io包中最重要的就是5个类和一个接口。5个类指的是File、OutputStream、InputStream、Writer、Reader;一个接口指的是Serializable.掌握了这些IO的核心操作那么对于Java中的IO体系也就有了一个初步的认识了。

Java I/O主要包括如下几个层次,包含三个部分:

    1.流式部分――IO的主体部分;

    2.非流式部分――主要包含一些辅助流式部分的类,如:File类、RandomAccessFile类和FileDescriptor等类;

    3.其他类--文件读取部分的与安全相关的类,如:SerializablePermission类,以及与本地操作系统相关的文件系统的类,如:FileSystem类和Win32FileSystem类和WinNTFileSystem类。

  主要的类如下:

      1. File(文件特征与管理):用于文件或者目录的描述信息,例如生成新目录,修改文件名,删除文件,判断文件所在路径等。

      2. InputStream(二进制格式操作):抽象类,基于字节的输入操作,是所有输入流的父类。定义了所有输入流都具有的共同特征。

      3. OutputStream(二进制格式操作):抽象类。基于字节的输出操作。是所有输出流的父类。定义了所有输出流都具有的共同特征。

      Java中字符是采用Unicode标准,一个字符是16位,即一个字符使用两个字节来表示。为此,JAVA中引入了处理字符的流。

4. Reader(文件格式操作):抽象类,基于字符的输入操作。

5. Writer(文件格式操作):抽象类,基于字符的输出操作。

6. RandomAccessFile(随机文件操作):它的功能丰富,可以从文件的任意位置进行存取(输入输出)操作。

     Java中IO流的体系结构如图:


Java io流路径 java中的io流_Java_05






5. 非流式文件类--File类


       在Java语言的java.io包中,由File类提供了描述文件和目录的操作与管理方法。但File类不是InputStream、OutputStream或Reader、Writer的子类,因为它不负责数据的输入输出,而专门用来管理磁盘文件与目录。



作用:File类主要用于命名文件、查询文件属性和处理文件目录。


  1. public    class   File   extends Objectimplements Serializable,Comparable  {}  

File类共提供了三个不同的构造函数,以不同的参数形式灵活地接收文件和目录名信息。构造函数:
1)File (String   pathname)   


     例:File  f1=new File("FileTest1.txt"); //创建文件对象f1,f1所指的文件是在当前目录下创建的FileTest1.txt
2)File (String  parent  ,  String child)


     例:File f2=new  File(“D:\\dir1","FileTest2.txt") ;//  注意:D:\\dir1目录事先必须存在,否则异常
3)File (File    parent  , String child)
     例:File  f4=new File("\\dir3");
          File  f5=new File(f4,"FileTest5.txt");  //在如果 \\dir3目录不存在使用f4.mkdir()先创建


        一个对应于某磁盘文件或目录的File对象一经创建, 就可以通过调用它的方法来获得文件或目录的属性。    


 


File类的方法:

(1) exists():测试磁盘中指定的文件或目录是否存在

(2) mkdir():创建文件对象指定的目录(单层目录)

(3) createNewFile():创建文件对象指定的文件

(4) list():返回目录中所有文件名字符串

(5)public boolean isFile( ) 判断是文件还是目录 

(6)public boolean isDirectory( )判断是文件还是目录

(7)public String getName( )返回文件名或目录名

( 8)public String getPath( )返回文件或目录的路径。

(9)public long length( )获取文件的长度

(10)public String[ ] list ( )将目录中所有文件名保存在字符串数组中返回。 

(1 1) public boolean renameTo( File newFile );    重命名文件

  (12) public void delete( );   删除文件

6. Java.IO流类库


1. io流的四个基本类


 java.io包中包含了流式I/O所需要的所有类。在java.io包中有四个基本类:InputStream、OutputStream及Reader、Writer类,它们分别处理字节流和字符流:


基本数据流的I/O


输入/输出

字节流

字符流

输入流

Inputstream

Reader

输出流

OutputStream

Writer



                                                                                   IO框架:


    

Java io流路径 java中的io流_java_06




Java中其他多种多样变化的流均是由它们派生出来的:


                                                     

Java io流路径 java中的io流_java_07

Java io流路径 java中的io流_数据_08

Java io流路径 java中的io流_java_09

Java io流路径 java中的io流_Java io流路径_10

JDK1.4版本开始引入了新I/O类库,它位于java.nio包中,新I/O类库利用通道和缓冲区等来提高I/O操作的效率。  在java.io包中,java.io.InputStream 表示字节输入流, java.io.OutputStream表示字节输出流,处于java.io包最顶层。这两个类均为抽象类,也就是说它们不能被实例化,必须生成子类之后才能实现一定的功能。


io流的具体分类:


一、按I/O类型来总体分类:

     1. Memory
1)从/向内存数组读写数据: CharArrayReader、 CharArrayWriter、ByteArrayInputStream、ByteArrayOutputStream    
 2)从/向内存字符串读写数据 StringReader、StringWriter、StringBufferInputStream

     2.Pipe管道  实现管道的输入和输出(进程间通信): PipedReader、PipedWriter、PipedInputStream、PipedOutputStream
     3.File 文件流。对文件进行读、写操作 :FileReader、FileWriter、FileInputStream、FileOutputStream
     4. ObjectSerialization 对象输入、输出 :ObjectInputStream、ObjectOutputStream
     5.DataConversion数据流 按基本数据类型读、写(处理的数据是Java的基本类型(如布尔型,字节,整数和浮点数)):DataInputStream、DataOutputStream
     6.Printing 包含方便的打印方法 :PrintWriter、PrintStream
     7.Buffering缓冲  在读入或写出时,对数据进行缓存,以减少I/O的次数:BufferedReader、BufferedWriter、BufferedInputStream、BufferedOutputStream
     8.Filtering 滤流,在数据进行读或写时进行过滤:FilterReader、FilterWriter、FilterInputStream、FilterOutputStream过
     9.Concatenation合并输入 把多个输入流连接成一个输入流 :SequenceInputStream 
    10.Counting计数  在读入数据时对行记数 :LineNumberReader、LineNumberInputStream
    11.Peeking Ahead 通过缓存机制,进行预读 :PushbackReader、PushbackInputStream
    12.Converting between Bytes and Characters 按照一定的编码/解码标准将字节流转换为字符流,或进行反向转换(Stream到Reader,Writer的转换类):InputStreamReader、OutputStreamWriter

二、按数据来源(去向)分类: 

1、File(文件): FileInputStream, FileOutputStream, FileReader, FileWriter 
2、byte[]:ByteArrayInputStream, ByteArrayOutputStream 
3、Char[]: CharArrayReader, CharArrayWriter 
4、String: StringBufferInputStream, StringReader, StringWriter 
5、网络数据流:InputStream, OutputStream, Reader, Writer 

7. 字节流InputStream/OutputStream

8. 字符流Writer/Reader

Java中字符是采用Unicode标准,一个字符是16位,即一个字符使用两个字节来表示。为此,JAVA中引入了处理字符的流。

1. Reader抽象类

    用于读取字符流的抽象类。子类必须实现的方法只有 read(char[], int, int) 和 close()。但是,多数子类将重写此处定义的一些方法,以提供更高的效率和/或其他功能。

Java io流路径 java中的io流_java_09

子类:

1) FileReader :与FileInputStream对应  ,主要用来读取字符文件,使用缺省的字符编码。

有三种构造函数: 

      (1)将文件名作为字符串 :FileReader f=new FileReader(“c:/temp.txt”); 
      (2)构造函数将File对象作为其参数。 
              File f=new file(“c:/temp.txt”); 
              FileReader f1=new FileReader(f); 
      (3)  构造函数将FileDescriptor对象作为参数 
            FileDescriptor() fd=new FileDescriptor() 
            FileReader f2=new FileReader(fd); 

2) CharArrayReader:与ByteArrayInputStream对应  


3) StringReader : 与StringBufferInputStream对应 


4) InputStreamReader :从输入流读取字节,在将它们转换成字符:Public inputstreamReader(inputstream is); 


5) FilterReader: 允许过滤字符流 : protected filterReader(Reader r); 


6) BufferReader :接受Reader对象作为参数,并对其添加字符缓冲器,使用readline()方法可以读取一行。 

     Public BufferReader(Reader r); 


  主要方法:

      (1)  public int read() throws IOException; //读取一个字符,返回值为读取的字符 



 /*读取len个字符,从数组cbuf[]的下标off处开始存放,返回值为实际读取的字符数量,该方法必须由子类实现*/ 




2. Writer抽象类

写入字符流的抽象类。子类必须实现的方法仅有 write(char[], int, int)、flush() 和 close()。但是,多数子类将重写此处定义的一些方法,以提供更高的效率和/或其他功能。 其子类如下:

Java io流路径 java中的io流_Java io流路径_10

 子类:

     1) FileWrite:与FileOutputStream对应  

   将字符类型数据写入文件,使用缺省字符编码和缓冲器大小。 
  Public FileWrite(file f); 
  将字符类型数据写入文件,使用缺省字符编码和缓冲器大小。 
  Public FileWrite(file f); 

  2)  chararrayWrite:与ByteArrayOutputStream对应 ,将字符缓冲器用作输出。       Public CharArrayWrite(); 
  3) PrintWrite:生成格式化输出 

      public PrintWriter(outputstream os); 

  4) filterWriter:用于写入过滤字符流 

      protected FilterWriter(Writer w); 


  5) PipedWriter:与PipedOutputStream对应  

      Public CharArrayWrite(); 


  6) PrintWrite:生成格式化输出 

      public PrintWriter(outputstream os); 

 

     7) StringWriter:无与之对应的以字节为导向的stream  



主要方法:

(1)  public void write(int c) throws IOException; //将整型值c的低16位写入输出流 
2)  public void write(char cbuf[]) throws IOException; //将字符数组cbuf[]写入输出流 
(3)  public abstract void write(char cbuf[],int off,int len) throws IOException; //将字符数组cbuf[]中的从索引为off的位置处开始的len个字符写入输出流 
(4)  public void write(String str) throws IOException; //将字符串str中的字符写入输出流 
(5)  public void write(String str,int off,int len) throws IOException; //将字符串str 中从索引off开始处的len个字符写入输出流 
(6)  flush( ) //刷空输出流,并输出所有被缓存的字节。 
(7)close() public abstract void close() throws IOException



3 .InputStream与Reader差别 OutputStream与Writer差别

InputStream和OutputStream类处理的是字节流,数据流中的最小单位是字节(8个bit)

Reader与Writer处理的是字符流,在处理字符流时涉及了字符编码的转换问题

Java io流路径 java中的io流_Java io流路径_13


Reader类能够将输入流中采用其他编码类型的字符转换为Unicode字符,然后在内存中为其分配内存

Writer类能够将内存中的Unicode字符转换为其他编码类型的字符,再写到输出流中。



9. FileWriter类(字符输出流类)

构造方法:

               FileWriter fw = new FileWriter(String fileName);//创建字符输出流类对象和已存在的文件相关联。文件不存在的话,并创建。

               如: FileWriter fw = new FileWriter("C:\\1.txt");

                 FileWriter fw = new FileWriter(String fileName,boolean append);//创建字符输出流类对象和已存在的文件相关联,并设置该该流对文件的操作是否为续写。

               如:FileWriter fw = new FileWriter("C:\\1.txt",ture); //表示在fw对文件再次写入时,会在该文件的结尾续写,并不会覆盖掉。

主要方法: 

        void write(String str)   //写入字符串。当执行完此方法后,字符数据还并没有写入到目的文件中去。此时字符数据会保存在缓冲区中。此时在使用刷新方法就可以使数据保存到目的文件中去。

        viod flush()                //刷新该流中的缓冲。将缓冲区中的字符数据保存到目的文件中去。

        viod close()               //关闭此流。在关闭前会先刷新此流的缓冲区。在关闭后,再写入或者刷新的话,会抛IOException异常。



10. 如何选择IO流

1)确定是数据源和数据目的(输入还是输出)

              源:输入流 InputStream Reader
              目的:输出流 OutputStream Writer

    2)明确操作的数据对象是否是纯文本

             是:字符流Reader,Writer
             否:字节流InputStream,OutputStream

    3)明确具体的设备。

是硬盘文件:File++:

             读取:FileInputStream,, FileReader, 

              写入:FileOutputStream,FileWriter
             是内存用数组

                  byte[]:ByteArrayInputStream, ByteArrayOutputStream
                  是char[]:CharArrayReader, CharArrayWriter
             是String:StringBufferInputStream(已过时,因为其只能用于String的每个字符都是8位的字符串), StringReader, StringWriter
             是网络用Socket流

是键盘:用System.in(是一个InputStream对象)读取,用System.out(是一个OutoutStream对象)打印

    4)是否需要转换流


            是,就使用转换流,从Stream转化为Reader,Writer:InputStreamReader,OutputStreamWriter 

提高效率

BufferedInputStream, BufferedOuputStream, BuffereaReader, BufferedWriter

    6)是否需要格式化输出


11. IOException异常类的子类


1.public class  EOFException :   非正常到达文件尾或输入流尾时,抛出这种类型的异常。

2.public class FileNotFoundException:   当文件找不到时,抛出的异常。

3.public class InterruptedIOException:   当I/O操作被中断时,抛出这种类型的异常。