`

Java IO的一般使用原则

 
阅读更多

Java IO的一般使用原则: 
 
一、按数据来源(去向)分类: 
 
1、是文件: FileInputStream, FileOutputStream, FileReader, FileWriter 
 
2、是byte[]:ByteArrayInputStream, ByteArrayOutputStream 
 
3、是Char[]: CharArrayReader, CharArrayWriter 
 
4、是String: StringBufferInputStream, StringReader, StringWriter 
 
5、网络数据流:InputStream, OutputStream, Reader, Writer 
 
二、按是否格式化输出分: 
 
1、要格式化输出:PrintStream, PrintWriter 
 
三、按是否要缓冲分: 
 
1、要缓冲:BufferedInputStream, BufferedOutputStream, BufferedReader, BufferedWriter 
 
四、按数据格式分: 
 
1、二进制格式(只要不能确定是纯文本的): InputStream, OutputStream及其所有带Stream结束的子类 
 
2、纯文本格式(含纯英文与汉字或其他编码方式);Reader, Writer及其所有带Reader, Writer的子类 
 
五、按输入输出分: 
 
1、输入:Reader, InputStream类型的子类 
 
2、输出:Writer, OutputStream类型的子类 
 
六、特殊需要: 
 
1、从Stream到Reader,Writer的转换类:InputStreamReader, OutputStreamWriter 
 
2、对象输入输出:ObjectInputStream, ObjectOutputStream 
 
3、进程间通信:PipeInputStream, PipeOutputStream, PipeReader, PipeWriter 
 
4、合并输入:SequenceInputStream 
 
5、更特殊的需要:PushbackInputStream, PushbackReader, LineNumberInputStream, LineNumberReader 
 
  
 
决定使用哪个类以及它的构造进程的一般准则如下(不考虑特殊需要): 
 
第一,考虑最原始的数据格式是什么:是否为文本? 
 
第二,是输入还是输出? 
 
第三,是否需要转换流:InputStreamReader, OutputStreamWriter? 
 
第四,数据来源(去向)是什么:文件?内存?网络? 
 
第五,是否要缓冲:bufferedReader (特别注明:一定要注意的是readLine()是否有定义,有什么比read, write更特殊的输入或输出方法) 
 
第六,是否要格式化输出:print? 
 
  
 
  
 
总结二: 
 
  
 
首先是java的IO。这破东西可真费事,I/O类库常使用”流(stream)”这种抽象。所谓”流”是一种能生成或接受数据的,代表数据的源和目标的对象。流把I/O设备内部的具体操作给隐藏起来了。 正如JDK文档所显示的,Java的I/O类库分成输入和输出两大部分。所有InputStream和Reader的派生类都有一个基本的,继承下来的,能读取单个或byte数组的read( )方法。同理,所有OutputStream和Writer的派生类都有一个基本的,能写入单个或byte数组的write( )方法。但通常情况下,你是不会去用这些方法的;它们是给其它类用的 —— 而后者会提供一些更实用的接口。因此,你很少会碰到只用一个类就能创建一个流的情形,实际上你得把多个对象叠起来,并以此来获取所需的功能。Java的流类库之所以会那么让人犯晕,最主要的原因就是”你必须为创建一个流而动用多个对象”。 
 
 
Java的IO类结构: 
      根接口是InputStream/OutputStream,充当数据源的IO类有FileInputStream/FileOutputStream,ByteArrayInputStream  / ByteArrayOutputStream  等,充当装饰功能的IO类有BufferedInputStream  /   BufferedOutputStream,DataInputStream   /   DataOutputStream等, 
     它们都是继承装饰接口FilterInputStream/FilterOutputStream。 
      使用IO时,首先创建一个数据源IO,然后根据需要的功能创建装饰类IO,其构造函数的参数为已创建的数据源IO。 
      我们以创建一个具有缓冲的文件输入流为例,假定需要从磁盘读取文件“C:\log.txt”: 
      // 创建一个FileInputStream: 
      FileInputStream fileInput = new FileInputStream(”C:\\log.txt”); 
      // 创建一个BufferedInputStream: 
      BufferedInputStream bufferedInput = new BufferedInputStream(fileInput); 
     // 现在得到的bufferedInput即是具有缓冲的文件输入流 
  或者进一步简写如下: 
     InputStream input = new BufferedInputStream(new FileInputStream(”C:\\log.txt”)); 
     // 现在得到的input即是具有缓冲的文件输入流 
 
 
  
 
java.io.Reader 和 java.io.InputStream 区别 
java.io.Reader 和 java.io.InputStream 组成了 Java 输入类。Reader 用于读入16位字符,也就是 Unicode 编码的字符;而 InputStream 用于读入 ASCII 字符和二进制数据。 
在 Java 中,有不同类型的 Reader 输入流对应于不同的数据源: 
    FileReader 用于从文件输入; 
    CharArrayReader 用于从程序中的字符数组输入; 
    StringReader 用于从程序中的字符串输入; 
    PipedReader 用于读取从另一个线程中的 PipedWriter 写入管道的数据。 
相应的也有不同类型的 InputStream 输入流对应于不同的数据源:FileInputStream,ByteArrayInputStream,StringBufferInputStream,PipedInputStream。另外,还有两种没有对应 Reader 类型的 InputStream 输入流: 
    Socket 用于套接字; 
    URLConnection 用于 URL 连接。 
这两个类使用 getInputStream() 来读取数据。 
相应的,java.io.Writer 和 java.io.OutputStream 也有类似的区别。 
 
 
1、Java技术支持两种数据类型的流 
InputStream和OutputStream:字节流。其它字节流都是InputStream或OutputStream的子类。 
Reader和 Writer:字符流。其它字符流都是Reader或Writer的子类。 
 
 
2、节点流 
Java 2 SDK中有三种基本类型的节点:文件(file)、内存(memory)、管道(pipe)。 
 
 
3、过程流 
过程流在其它流之上,完成排序、变换等操作。过程流也被称做过滤流。 
当你需要改变输入流的原始数据时,你可以将一个过滤输入流连接到一个原始的输入流上。 
用过滤流将原始数据变换成你需要的格式。   
 
 
4、基本字节流类 
4.1、FileInputStream和FileOutputStream 
这两个节点流用来操纵磁盘文件。这些类的构造函数允许你指定它们所连接的文件。 
要构造一个FileInputStream,所关联的文件必须存在而且是可读的。 
如果你要构造一个FileOutputStream而输出文件已经存在,则它将被覆盖。 
FileInputStream infile = new FileInputStream(”myfile.dat”); 
FileOutputStream outfile = new FileOutputStream(”results.dat”); 
4.1、 BufferInputStream和BufferOutputStream 
这些是过滤器流,它们可以提高I/O操作的效率。 
4.3、 PipedInputStream和PipedOutputStream 
管道流用来在线程间进行通信。一个线程的PipedInputStream对象从另一个线程的PipedOutputStream对象读取输入。 
要使管道流有用,必须有一个输入方和一个输出方。 
4.4、 DataInputStream和DataOutputStream 
这些过滤器通过流来读写Java基本类 
 
 
5、 基本字符流类 
图阐述了Reader和Writer字符流的体系结构。 
5.1、InputStreamReader 和 OutputStreamWriter 
用于字节流与字符流之间的转换接口。 
当你构造一个InputStreamReader或OutputStreamWriter时,转换规则定义了16位Unicode和其它平台的特定表示之间的转换。 
InputStreamReader从一个数据源读取字节,并自动将其转换成Unicode字符。 
如果你特别声明,InputStreamReade会将字节流转换成其它种类的字符流。 
OutputStreamWriter将字符的Unicode编码写到输出流,如果你的使用的不是Unicode字符,OutputStreamWriter会将你的字符编码转换成Unicode编码。 
5.2.、缓冲读者和作者 
因为在各种格式之间进行转换和其它I/O操作很类似,所以在处理大块数据时效率最高。 
在InputStreamReader和OutputStreamWriter的结尾链接一个BufferedReader和BufferedWriter是一个好主意。 
记住对BufferedWriter使用flush()方法。 
5.3、 使用其它字符转换 
如果你需要从一个非本地(例如,从连接到一个不同类型的机器的网络连接读取)的字符编码读取输入, 
你可以象下面这个程序那样,使用显式的字符编码构造ir=new InputStreamReader(System.in,  “8859_1″); 
注:如果你通过网络连接读取字符,就应该使用这种形式。 
否则,你的程序会总是试图将所读取的字符当作本地表示来进行转换,而这并不总是正确的。ISO 8859-1是映射到ASCII的Latin-1编码模式。 
 
 
6、 对象串行化 
java.io.Serializable接口支持将一个Java技术对象存放到一个流中。 
将一个对象存放到某种类型的永久存储器上称为”保持”。 
如果一个对象可以被存放到磁盘或磁带上,或者可以发送到另外一台机器并存放到存储器或磁盘上,那么这个对象就被称为可保持的。 
java.io.Serializable接口没有任何方法,它只作为一个”标记”,用来表明实现了这个接口的类可以串行化。 
类中没有实现Serializable接口的对象不能被保持。 
// 文件实现追加: 
// 其中的FileWriter()中的第二个参数的含义是:是否在文件中追加内容 
PrintWriter out = new PrintWriter(new FileWriter(logFileName, true), true); 
Java读写文件最常用的类是FileInputStream/FileOutputStream和FileReader/FileWriter。 
其中FileInputStream和FileOutputStream是基于字节流的,常用于读写二进制文件。 
读写字符文件建议使用基于字符的FileReader和FileWriter,省去了字节与字符之间的转换。 
但这两个类的构造函数默认使用系统的编码方式,如果文件内容与系统编码方式不一致,可能会出现乱码。 
在这种情况下,建议使用FileReader和FileWriter的父类:InputStreamReader/OutputStreamWriter, 
它们也是基于字符的,但在构造函数中可以指定编码类型:InputStreamReader(InputStream in, Charset cs) 和OutputStreamWriter(OutputStream out, Charset cs)。   
 
  
 
  
 
// 读写文件的编码: 
InputStreamReader r = new InputStreamReader(new FileInputStream(fileName), “utf-8″); 
OutputStreamWriter out = new OutputStreamWriter(new FileOutputStream(fileName),”utf-8″); 
 
/**
三种IO性能比较:
在读写一个10k文件的时候,三种方式的耗时如下:
InputStreamReader And OutputStreamWriter : 63ms (可以设置文件的编码,如果不用buffer)
BufferedReader And BufferedWriter : 31ms 
BufferedInputStream And BufferedOutputStream : 16ms
*/ 
 
/**
* Description: Test the java IO’s efficiency
* Author: AllanCao
* Date: 2007-02-18
*/ 
import java.io.*; 
 
/**
* using the InputStreamReader And OutputStreamWriter
*/ 
class EncoderRW { 
public static String read(String fileName) throws IOException { 
  StringBuffer sb = new StringBuffer(); 
  /*此处读文件时用了buffer,如果不用,性能损失一倍*/ 
  BufferedReader in = new BufferedReader(new InputStreamReader(new FileInputStream(fileName), “utf-8″)); 
  String s; 
  while((s = in.readLine()) != null) { 
    sb.append(s); 
    sb.append(”\n”); 
  } 
  in.close(); 
  return sb.toString(); 

public void write(String fileName, String text) throws IOException { 
  OutputStreamWriter out = new OutputStreamWriter(new FileOutputStream(fileName),”utf-8″); 
  out.write(text); 
  out.flush(); 
  out.close(); 


 
/**
* using the BufferedReader And BufferedWriter
*/ 
class WriterReader { 
public String read(String fileName) throws IOException { 
  StringBuffer sb = new StringBuffer(); 
  BufferedReader in = new BufferedReader(new FileReader(fileName)); 
  String s; 
  while((s = in.readLine()) != null) { 
    sb.append(s); 
    sb.append(”\n”); 
  } 
  in.close(); 
  return sb.toString(); 

public void write(String fileName, String text) throws IOException { 
  PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(fileName))); 
  out.print(text); 
  out.close(); 


 
/**
* using the BufferedInputStream And BufferedOutputStream
*/ 
class BufferedStream{ 
public byte[] read(String fileName) throws IOException { 
  BufferedInputStream remoteBIS = new BufferedInputStream(new FileInputStream(fileName)); 
  ByteArrayOutputStream baos = new ByteArrayOutputStream(10240); 
  byte[] buf = new byte[1024]; 
  int bytesRead = 0; 
  while(bytesRead >= 0) 
  { 
   baos.write(buf, 0, bytesRead); 
   bytesRead = remoteBIS.read(buf); 
  } 
  byte[] content = baos.toByteArray(); 
  return content; 

public void write(String fileName, byte[] content)  throws IOException { 
  BufferedOutputStream out = new BufferedOutputStream(new FileOutputStream(fileName)); 
  out.write(content); 
  out.flush(); 
  out.close(); 


 
public class TestIO 

public static void main(String[] args)throws IOException { 
  long currentTime = System.currentTimeMillis() ; 
  EncoderRW rw = new EncoderRW(); 
  rw.write(”index.dat”,rw.read(”FileUtil.java”)); 
  System.out.println(”cost time:” + Long.toString(System.currentTimeMillis()-currentTime) + ” ms”); 
 
  currentTime = System.currentTimeMillis() ; 
  WriterReader wr = new WriterReader(); 
  wr.write(”index.dat”,wr.read(”FileUtil.java”)); 
  System.out.println(”cost time:” + Long.toString(System.currentTimeMillis()-currentTime) + ” ms”); 
 
  currentTime = System.currentTimeMillis() ; 
  BufferedStream bf = new BufferedStream(); 
  bf.write(”index.dat”,bf.read(”FileUtil.java”)); 
  System.out.println(”cost time:” + Long.toString(System.currentTimeMillis()-currentTime) + ” ms”); 


http://dinghaoliang.blog.163.com/blog/static/12654071420106921425505/ 
分享到:
评论

相关推荐

    java IO详解课件

    - **Java I/O系统的设计原则**:Java I/O系统遵循面向对象的设计思想,提供了一系列类和接口来支持各种类型的输入输出操作。这些类和接口按照功能的不同被组织在`java.io`包中。 - **设计模式的应用**:Java I/O...

    java小项目练习IO实现

    总之,这个Java小项目“Olymplic Games Project(IO版)”是一个很好的实践平台,它涵盖了Java IO流的使用,面向对象的设计原则,以及集合框架的应用。通过实际操作,你可以更深入地理解这些核心概念,并提升自己的...

    Java_IO完全总结

    Java IO系统的设计原则之一是高度模块化和可扩展性,通过抽象出一系列接口来支持各种不同类型的I/O操作。例如,`InputStream`和`OutputStream`接口分别用于读取和写入数据,它们构成了Java IO系统的核心基础。此外,...

    java.io包详细解说

    【Java IO包详解】 ...理解这个模式对于深入掌握Java IO非常重要,它不仅帮助我们更好地理解和使用Java提供的IO类,还可以启发我们在其他领域中应用类似的设计原则,以实现更加可扩展和灵活的代码。

    java_io详解

    Java IO类结构的设计遵循了装饰者模式的原则,根接口为`InputStream`和`OutputStream`。这些接口定义了基本的输入输出操作。具体的数据源类如`FileInputStream`、`FileOutputStream`、`ByteArrayInputStream`、`...

    使用java swing、javaio基本操作以及mysql的农场小游戏

    《使用Java Swing、Java IO基本操作以及MySQL的农场小游戏》 本项目是一个针对初学者的课程设计,旨在帮助学生深入理解和应用Java编程语言,特别是Java Swing库用于图形用户界面(GUI)开发,Java IO用于文件操作,...

    JAVA IO流小结

    五、输入/输出流的使用原则 *凡是跨出虚拟机边界的资源都要求程序员自己关闭,不要指望垃圾回收。 *在关闭流时只用关闭最外层的流。 *I/O 流是一类很宝贵的资源,使用完后必须调用close()方法关闭流并释放资源。 ...

    java.io详解

    #### 三、Java IO流使用原则 1. 如果处理的数据是二进制内容,优先选择字节流。 2. 如果处理的是文本数据,优先选择字符流。 3. 当需要访问文件内容时,应结合使用`File`类和相应的输入输出流。 #### 四、File类...

    新版Java_IO精讲(推荐版)

    ### 新版Java_IO精讲(推荐版) #### Java I/O简介 Java I/O(输入/输出)是Java编程语言中非常重要的组成部分,主要用于处理文件、控制台以及网络连接等场景下的数据读写操作。Java I/O支持两种基本的数据类型:...

    JAVA IO API使用详解

    2. 使用原则: - 首先确定数据格式:是二进制还是文本。 - 确定数据流向:是输入还是输出。 - 如果需要转换数据格式,使用InputStreamReader或OutputStreamWriter。 - 确定数据源或目标:是文件、内存还是网络。...

    Java-从Decorator,Adapter模式看Java/IO

    Java的IO库遵循了输入-输出对称性和byte-char对称性原则,确保了在处理输入流和输出流时的统一性。输入-输出对称性体现在InputStream和OutputStream,Reader和Writer这两对类上,它们各自构成了平行的等级结构,分别...

    java编程规范编程原则

    import java.io.*; import java.util.Observable; import hotlava.util.Application; ``` 3. **类定义**:每个类都应该有清晰的注释,说明类的功能和用途。例如: ```java /** * A class representing a set ...

    JavaIO性能分析[定义].pdf

    Java IO性能分析主要关注如何有效地处理输入和输出操作,这对于任何软件开发,特别是涉及大量数据交换的应用至关重要。本文档深入探讨了Java中的IO性能优化,包括底层和高级I/O问题。 首先,Java提供了两种基本的...

    JAVA IO-(FileBuffered的InputOutputStream的基本操作)

    在Java编程语言中,IO(Input/Output)流是处理数据输入和输出的关键部分,而FileInputStream和FileOutputStream是Java IO中的基础类,用于读写文件。为了提高效率,我们通常会结合BufferedInputStream和...

    java宠物管理

    在这个项目中,开发者可能使用了Java的标准库,如集合框架、IO流和多线程等,来实现系统的功能。 描述中提到的"导入工程"表明这是一个完整的Java项目,可能包含多个源代码文件、资源文件以及配置文件。通常,这样的...

    50丨装饰器模式:通过剖析JavaIO类库源码学习装饰器模式1

    在Java IO类库中,装饰器模式被广泛使用,以灵活地扩展类的功能,而避免了使用继承带来的复杂性。装饰器模式的核心在于,它定义了一个与组件接口相同的接口,因此可以在运行时动态地将责任附加到对象上。 在Java IO...

    可以看成是java.io的扩展。它提供了文件阅读功能,以及一些使用的方法

    Java IO 是Java平台的核心部分,用于处理输入/输出操作,如读写文件、网络通信等。而Apache Commons IO是Java IO的一个强大补充,它提供了一系列实用工具类,扩展了Java的标准IO功能,使得开发人员在处理文件、流、...

    JAVA-IO详解.txt

    Java IO的设计遵循了面向对象的原则,并提供了丰富的接口来满足各种不同的需求。 #### 二、字节流(Byte Stream) 字节流是Java中最基本的数据传输方式之一,适用于所有类型的数据传输,包括文本和二进制数据。...

    java版学生成绩管理系统

    一般情况下,可能会选择使用`ArrayList`或`HashMap`等集合类来存储学生对象。`ArrayList`适合于顺序访问,而`HashMap`则提供了基于键值对的快速查找,这在查询学生信息时非常有用。如果需要按照总分排名,可以使用`...

Global site tag (gtag.js) - Google Analytics