首页 > 其他 > 详细

I/O流

时间:2015-12-28 23:22:48      阅读:244      评论:0      收藏:0      [点我收藏+]

一、什么是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。

 


 

二、数据流的基本概念

   在电脑上的数据有三种存储方式,一种是外存,一种是内存,一种是缓存。比如电脑上的硬盘,磁盘,U盘等都是外存,在电脑上有内存条,缓存是在CPU里面的。外存的存储量最大,其次是内存,最后是缓存,但是外存的数据的读取最慢,其次是内存,缓存最快。这里总结从外存读取数据到内存以及将数据从内存写到外存中。对于内存和外存的理解,我们可以简单的理解为容器,即外存是一个容器,内存又是另外一个容器。那又怎样把放在外存这个容器内的数据读取到内存这个容器以及怎么把内存这个容器里的数据存到外存中呢?

     在Java类库中,IO部分的内容是很庞大的,因为它涉及的领域很广泛:

         标准输入输出,文件的操作,网络上的数据流,字符串流,对象流,zip文件流等等,java中将输入输出抽象称为流,就好像水管,将两个容器连接起来。将数据冲外存中读取到内存中的称为输入流,将数据从内存写入外存中的称为输出流。

    流是一个很形象的概念,当程序需要读取数据的时候,就会开启一个通向数据源的流,这个数据源可以是文件,内存,或是网络连接。类似的,当程序需要写入数据的时候,就会开启一个通向目的地的流。

  1) 数据流

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

技术分享

  2) 输入流(Input  Stream):

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

技术分享

  3) 输出流

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

  4) 数据流分类:

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

三、标准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             System.out.println("args[" + i + "] is <" + args[i] + ">");
5         }
6     }
7 }

  运行命令:java Java C VB

  运行结果:

1 args[0] is <Java>
2 args[1] is <C>
3 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。

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

 结果为:

技术分享

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         InputStreamReader ir = new InputStreamReader(System.in);
10         BufferedReader in = new BufferedReader(ir);
11         System.out.println("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                 System.out.println("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 }
View Code

四、java.IO层次体系结构

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

技术分享

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

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

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

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

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

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

五、非流式文件类-File类

  File类提供了描述文件和目录的操作与管理方法。但File类不是InputStream、OutputStream或Reader、Writer的子类,因为它不负责数据的输入输出,而专门用来管理磁盘文件与目录。File类作用:File类主要用于命名文件、查询文件属性和处理文件目录。

 1 一个对应于某磁盘文件或目录的File对象一经创建, 就可以通过调用它的方法来获得文件或目录的属性。  
 2     1)public boolean exists( ) 判断文件或目录是否存在 
 3     2)public boolean isFile( ) 判断是文件还是目录
 4     3)public boolean isDirectory( ) 判断是文件还是目录   
 5     4)public String getName( ) 返回文件名或目录名    
 6     5)public String getPath( ) 返回文件或目录的路径。   
 7     6)public long length( ) 获取文件的长度    
 8     7)public String[ ] list ( ) 将目录中所有文件名保存在字符串数组中返回。
 9 
10   
11 File类中还定义了一些对文件或目录进行管理、操作的方法,常用的方法有:
12     1) public boolean renameTo( File newFile );重命名文件   
13     2) public void delete( );   删除文件    
14     3) public boolean mkdir( ); 创建目录
技术分享
 1 package io;
 2 import java.io.*;
 3 public class Test {
 4     public static void main(String[] args)throws IOException{
 5         File dir=new File("C:/Users/Maria/Desktop");
 6         File f1=new File(dir,"1.txt");
 7         File f2=new File(dir,"2.java");
 8         // 文件对象创建后,指定的文件或目录不一定物理上存在 
 9         if(!dir.exists()){
10             dir.mkdir();//创建文件夹
11         }
12         if(!f1.exists()){
13             f1.createNewFile();//创建新文件
14         }
15         if(!f2.exists()){
16             f2.createNewFile();//创建新文件
17         }        
18         System.out.println("f1‘s AbsolutePath= "+f1.getAbsolutePath());//显示出绝对路径
19         System.out.println("f1 Canread= "+f1.canRead());
20         System.out.println("f1‘s len= "+f1.length());
21         String[] FL;
22         int count=0;
23         FL=dir.list();
24         for(int i=0;i<FL.length;i++){
25             count++;
26             System.out.println(FL[i]+" is in C:/Users/Maria/Dekstop");//显示出桌面所有的文件和文件夹
27         }
28         System.out.println("there are "+count+" file in// C:/Users/Maria/Dekstop");//统计文件和文件夹的个数
29     }
30 }
View Code

  结果为:

技术分享
 1 f1‘s AbsolutePath= C:\Users\Maria\Desktop\1.txt
 2 f1 Canread= true
 3 f1‘s len= 734
 4 1.txt is in C:/Users/Maria/Dekstop
 5 2.java is in C:/Users/Maria/Dekstop
 6 desktop.ini is in C:/Users/Maria/Dekstop
 7 娱乐 is in C:/Users/Maria/Dekstop
 8 文件 is in C:/Users/Maria/Dekstop
 9 浏览器 is in C:/Users/Maria/Dekstop
10 系统 is in C:/Users/Maria/Dekstop
11 聊天 is in C:/Users/Maria/Dekstop
12 软件 is in C:/Users/Maria/Dekstop
13 there are 9 file in// C:/Users/Maria/Dekstop
View Code

 


六、Java.IO流类库

  在java.io包中有四个基本类:InputStream、OutputStream及Reader、Writer类,它们分别处理字节流和字符流:

技术分享

 技术分享

  按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

 


七、字节流InputStream/OutputStream

 

1、InputStream抽象类

  InputStream 为字节输入流,它本身为一个抽象类,必须依靠其子类实现各种功能,此抽象类是表示字节输入流的所有类的超类。 继承自InputStream  的流都是向程序中输入数据的,且数据单位为字节(8bit);
 InputStream是输入字节数据用的类,所以InputStream类提供了3种重载的read方法.Inputstream类中的常用方法:
     (1) public abstract int read( ):读取一个byte的数据,返回值是高位补0的int类型值。若返回值=-1说明没有读取到任何字节读取工作结束。
     (2) public int read(byte b[ ]):读取b.length个字节的数据放到b数组中。返回值是读取的字节数。该方法实际上是调用下一个方法实现的 
     (3) public int read(byte b[ ], int off, int len):从输入流中最多读取len个字节的数据,存放到偏移量为off的b数组中。 
     (4) public int available( ):返回输入流中可以读取的字节数。注意:若输入阻塞,当前线程将被挂起,如果InputStream对象调用这个方法的话,它只会返回0,这个方法必须由继承InputStream类的子类对象调用才有用
       (5) public long skip(long n):忽略输入流中的n个字节,返回值是实际忽略的字节数, 跳过一些字节来读取
     (6) public int close( ) :我们在使用完后,必须对我们打开的流进行关闭

2.OutputStream抽象类

  流结束的判断:方法read()的返回值为-1时;readLine()的返回值为null时。

OutputStream提供了3个write方法来做数据的输出,这个是和InputStream是相对应的。 
  1. public void write(byte b[ ]):将参数b中的字节写到输出流。 
  2. public void write(byte b[ ], int off, int len) :将参数b的从偏移量off开始的len个字节写到输出流。 
  3. public abstract void write(int b) :先将int转换为byte类型,把低字节写入到输出流中 
  4. public void flush( ) : 将数据缓冲区中数据全部输出,并清空缓冲区。 
  5. public void close( ) : 关闭输出流并释放与流相关的系统资源。 

3. 文件输入流: FileInputStream类

   FileInputStream可以使用read()方法一次读入一个字节,并以int类型返回,或者是使用read()方法时读入至一个byte数组,byte数组的元素有多少个,就读入多少个字节。在将整个文件读取完成或写入完毕的过程中,这么一个byte数组通常被当作缓冲区,因为这么一个byte数组通常扮演承接数据的中间角色。

技术分享

  使用方法:

    File fin=new File("d:/abc.txt");

    FileInputStream in=new FileInputStream( fin);或者是

    FileInputStream  in=new  FileInputStream(“d: /abc.txt”);

  例如:将InputFromFile.java的程序的内容显示在显示器上。即|:

  技术分享

技术分享
 1 package io;
 2 import java.io.*;
 3 public class Test2 {
 4     public static void main(String[] args)throws IOException{
 5         try{
 6             FileInputStream rf=new FileInputStream("C:/Users/Maria/Desktop/1.txt");
 7             int n=512;
 8             byte buffer[]=new byte[n];
 9             while((rf.read(buffer,0,n)!=-1)&&(n>0)){
10                 System.out.println(new String(buffer));
11             }
12             System.out.println();
13             rf.close();
14         }
15         catch(IOException e){
16             e.printStackTrace();
17         }
18     }
19 }
View Code
技术分享
 1 File类中还定义了一些对文件或目录进行管理、操作的方法,常用的方法有:
 2     1) public boolean renameTo( File newFile );重命名文件   
 3     2) public void delete( );   删除文件    
 4     3) public boolean mkdir( ); 创建目录olean isDirectory( ) 判断是文件还是目录   
 5     4)public String getName( ) 返回文件名或目录名    
 6     5)public String getPath( ) 返回文件或目录的路径。   
 7     6)public long length( ) 获取文件的长度    
 8     7)public String[ ] list ( ) 将目录中所有文件名保存在字符串数组中返回。
 9 
10   
View Code

4.文件输出流:FileOutputStream类

  从内存区读数据入文件。创建一个文件流对象的方法:

    FileOutputStream out=new FileOutputStream(“d:/myjava/write.txt "); 

  例子1:使用键盘输入一段文章,将文章保存在write.txt中

技术分享

技术分享
 1 package io;
 2 import java.io.*;
 3 public class Test2 {
 4     public static void main(String[] args)throws IOException{
 5         try{
 6             System.out.println("please Input from Keyboard:");
 7             int count,n=512;
 8             byte buffer[]=new byte[n];
 9             count=System.in.read(buffer);
10             FileOutputStream rf=new FileOutputStream("C:/Users/Maria/Desktop/1.txt");            
11             rf.write(buffer,0,count);
12             rf.close();
13             //while((rf.read(buffer,0,n)!=-1)&&(n>0)){
14             System.out.println("Save to the 1.txt");
15         }
16         catch(IOException e){
17             System.out.println("ERROR");
18         }
19     }
20 }
View Code

   结果为:

技术分享
1 please Input from Keyboard:
2 NIHAO MA 
3 Save to the 1.txt
View Code

   例子2:利用程序将文件file1.txt 拷贝到file2.txt中。

技术分享
 1 package io;
 2 import java.io.*;
 3 public class Test2 {
 4     public static void main(String[] args)throws IOException{
 5         try{
 6             FileInputStream rf=new FileInputStream("C:/Users/Maria/Desktop/1.txt");            
 7             FileOutputStream wf=new FileOutputStream("C:/Users/Maria/Desktop/2.java");            
 8             int c;
 9             while((c=rf.read())!=-1){
10                 wf.write(c);
11             }
12             System.out.println("文件拷贝成功!");
13             rf.close();
14             wf.close();
15         
16         }
17         catch(IOException e){
18             System.out.println("ERROR");
19         }
20     }
21 }
View Code

5. 缓冲输入输出流 BufferedInputStream/ BufferedOutputStream

技术分享

  计算机访问外部设备非常耗时。访问外存的频率越高,造成CPU闲置的概率就越大。为了减少访问外存的次数,应该在一次对外设的访问中,读写更多的数据。为此,除了程序和流节点间交换数据必需的读写机制外,还应该增加缓冲机制。缓冲流就是每一个数据流分配一个缓冲区,一个缓冲区就是一个临时存储数据的内存。这样可以减少访问硬盘的次数,提高传输效率。

  BufferedInputStream:当向缓冲流写入数据时候,数据先写到缓冲区,待缓冲区写满后,系统一次性将数据发送给输出设备。

  BufferedOutputStream :当从向缓冲流读取数据时候,系统先从缓冲区读出数据,待缓冲区为空时,系统再从输入设备读取数据到缓冲区。

  1)将文件读入内存

    将BufferedInputStream与FileInputStream相接。

    FileInputStream in=new  FileInputStream( “file1.txt ” );

    BufferedInputStream bin=new  BufferedInputStream( in);

  2)将内存写入文件

    将BufferOutputStream与FileOutputStream相接。

    FileOutputStream out=new FileOutputStream(“file1.txt”);

    BufferedOutputStream  bin=new BufferedInputStream(out);

  3)从键盘上读入流读到内存

    将BufferedReader与标准的数据流相接

    InputStreamReader sin=new InputStreamReader(System.in);

    BufferedReader bin=new BufferedReader(sin);

技术分享
 1 package io;
 2 import java.io.*;
 3 public class Test2 {
 4     public static void main(String[] args)throws IOException{
 5         try{System.out.println("please Input from  Keyboard"); 
 6             InputStreamReader rf=new InputStreamReader(System.in);            
 7             BufferedReader wf = new BufferedReader(rf);
 8             FileWriter out=new FileWriter("C:/Users/Maria/Desktop/2.java");            
 9             BufferedWriter bout = new BufferedWriter(out); 
10             String s;
11             while((s=wf.readLine()).length()>0){
12                 bout.write(s,0,s.length());
13             }
14             System.out.println("文件拷贝成功!");
15         }
16         catch(IOException e){
17             System.out.println("ERROR");
18         }
19     }
20 }
View Code

 

I/O流

原文:http://www.cnblogs.com/liuzhongfeng/p/5084309.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!