JAVA IO流

  1. Java IO的通常使用原则:  
  2.   
  3. 1、按数据来源(去向)分类:  
  4.   
  5. 一、是文件: FileInputStream, FileOutputStream, FileReader, FileWriter  
  6.   
  7. 二、是byte[]:ByteArrayInputStream, ByteArrayOutputStream  
  8.   
  9. 三、是Char[]: CharArrayReader, CharArrayWriter  
  10.   
  11. 四、是String: StringBufferInputStream, StringReader, StringWriter  
  12.   
  13. 五、网络数据流:InputStream, OutputStream, Reader, Writer  
  14.   
  15. 2、按是否格式化输出分:  
  16.   
  17. 一、要格式化输出:PrintStream, PrintWriter  
  18.   
  19. 3、按是否要缓冲分:  
  20.   
  21. 一、要缓冲:BufferedInputStream, BufferedOutputStream, BufferedReader, BufferedWriter  
  22.   
  23. 4、按数据格式分:  
  24.   
  25. 一、二进制格式(只要不能肯定是纯文本的): InputStream, OutputStream及其全部带Stream结束的子类  
  26.   
  27. 二、纯文本格式(含纯英文与汉字或其余编码方式);Reader, Writer及其全部带Reader, Writer的子类  
  28.   
  29. 5、按输入输出分:  
  30.   
  31. 一、输入:Reader, InputStream类型的子类  
  32.   
  33. 二、输出:Writer, OutputStream类型的子类  
  34.   
  35. 6、特殊须要:  
  36.   
  37. 一、从Stream到Reader,Writer的转换类:InputStreamReader, OutputStreamWriter  
  38.   
  39. 二、对象输入输出:ObjectInputStream, ObjectOutputStream  
  40.   
  41. 三、进程间通讯:PipeInputStream, PipeOutputStream, PipeReader, PipeWriter  
  42.   
  43. 四、合并输入:SequenceInputStream  
  44.   
  45. 五、更特殊的须要:PushbackInputStream, PushbackReader, LineNumberInputStream, LineNumberReader  
  46.   
  47.    
  48.   
  49. 决定使用哪一个类以及它的构造进程的通常准则以下(不考虑特殊须要):  
  50.   
  51. 第一,考虑最原始的数据格式是什么:是否为文本?  
  52.   
  53. 第二,是输入仍是输出?  
  54.   
  55. 第三,是否须要转换流:InputStreamReader, OutputStreamWriter?  
  56.   
  57. 第四,数据来源(去向)是什么:文件?内存?网络?  
  58.   
  59. 第五,是否要缓冲:bufferedReader (特别注明:必定要注意的是readLine()是否有定义,有什么比read, write更特殊的输入或输出方法)  
  60.   
  61. 第六,是否要格式化输出:print?  
  62.   
  63.    
  64.   
  65.    
  66.   
  67. 总结二:  
  68.   
  69.    
  70.   
  71. 首先是java的IO。这破东西可真费事,I/O类库常使用”流(stream)”这种抽象。所谓”流”是一种能生成或接受数据的,表明数据的源和目标的对象。流把I/O设备内部的具体操做给隐藏起来了。 正如JDK文档所显示的,Java的I/O类库分红输入和输出两大部分。全部InputStream和Reader的派生类都有一个基本的,继承下来的,能读取单个或byte数组的read( )方法。同理,全部OutputStream和Writer的派生类都有一个基本的,能写入单个或byte数组的write( )方法。但一般状况下,你是不会去用这些方法的;它们是给其它类用的 —— 然后者会提供一些更实用的接口。所以,你不多会碰到只用一个类就能建立一个流的情形,实际上你得把多个对象叠起来,并以此来获取所需的功能。Java的流类库之因此会那么让人犯晕,最主要的缘由就是”你必须为建立一个流而动用多个对象”。  
  72.   
  73.   
  74. Java的IO类结构:  
  75.       根接口是InputStream/OutputStream,充当数据源的IO类有FileInputStream/FileOutputStream,ByteArrayInputStream  / ByteArrayOutputStream  等,充当装饰功能的IO类有BufferedInputStream  /   BufferedOutputStream,DataInputStream   /   DataOutputStream等,  
  76.      它们都是继承装饰接口FilterInputStream/FilterOutputStream。  
  77.       使用IO时,首先建立一个数据源IO,而后根据须要的功能建立装饰类IO,其构造函数的参数为已建立的数据源IO。  
  78.       咱们以建立一个具备缓冲的文件输入流为例,假定须要从磁盘读取文件“C:\log.txt”:  
  79.       // 建立一个FileInputStream:  
  80.       FileInputStream fileInput = new FileInputStream(”C:\\log.txt”);  
  81.       // 建立一个BufferedInputStream:  
  82.       BufferedInputStream bufferedInput = new BufferedInputStream(fileInput);  
  83.      // 如今获得的bufferedInput便是具备缓冲的文件输入流  
  84.   或者进一步简写以下:  
  85.      InputStream input = new BufferedInputStream(new FileInputStream(”C:\\log.txt”));  
  86.      // 如今获得的input便是具备缓冲的文件输入流  
  87.   
  88.   
  89.    
  90.   
  91. java.io.Reader 和 java.io.InputStream 区别  
  92. java.io.Reader 和 java.io.InputStream 组成了 Java 输入类。Reader 用于读入16位字符,也就是 Unicode 编码的字符;而 InputStream 用于读入 ASCII 字符和二进制数据。  
  93. 在 Java 中,有不一样类型的 Reader 输入流对应于不一样的数据源:  
  94.     FileReader 用于从文件输入;  
  95.     CharArrayReader 用于从程序中的字符数组输入;  
  96.     StringReader 用于从程序中的字符串输入;  
  97.     PipedReader 用于读取从另外一个线程中的 PipedWriter 写入管道的数据。  
  98. 相应的也有不一样类型的 InputStream 输入流对应于不一样的数据源:FileInputStream,ByteArrayInputStream,StringBufferInputStream,PipedInputStream。另外,还有两种没有对应 Reader 类型的 InputStream 输入流:  
  99.     Socket 用于套接字;  
  100.     URLConnection 用于 URL 链接。  
  101. 这两个类使用 getInputStream() 来读取数据。  
  102. 相应的,java.io.Writer 和 java.io.OutputStream 也有相似的区别。  
  103.   
  104.   
  105. 一、Java技术支持两种数据类型的流  
  106. InputStream和OutputStream:字节流。其它字节流都是InputStream或OutputStream的子类。  
  107. Reader和 Writer:字符流。其它字符流都是Reader或Writer的子类。  
  108.   
  109.   
  110. 二、节点流  
  111. Java 2 SDK中有三种基本类型的节点:文件(file)、内存(memory)、管道(pipe)。  
  112.   
  113.   
  114. 三、过程流  
  115. 过程流在其它流之上,完成排序、变换等操做。过程流也被称作过滤流。  
  116. 当你须要改变输入流的原始数据时,你能够将一个过滤输入流链接到一个原始的输入流上。  
  117. 用过滤流将原始数据变换成你须要的格式。    
  118.   
  119.   
  120. 四、基本字节流类  
  121. 4.一、FileInputStream和FileOutputStream  
  122. 这两个节点流用来操纵磁盘文件。这些类的构造函数容许你指定它们所链接的文件。  
  123. 要构造一个FileInputStream,所关联的文件必须存在并且是可读的。  
  124. 若是你要构造一个FileOutputStream而输出文件已经存在,则它将被覆盖。  
  125. FileInputStream infile = new FileInputStream(”myfile.dat”);  
  126. FileOutputStream outfile = new FileOutputStream(”results.dat”);  
  127. 4.一、 BufferInputStream和BufferOutputStream  
  128. 这些是过滤器流,它们能够提升I/O操做的效率。  
  129. 4.三、 PipedInputStream和PipedOutputStream  
  130. 管道流用来在线程间进行通讯。一个线程的PipedInputStream对象从另外一个线程的PipedOutputStream对象读取输入。  
  131. 要使管道流有用,必须有一个输入方和一个输出方。  
  132. 4.四、 DataInputStream和DataOutputStream  
  133. 这些过滤器经过流来读写Java基本类  
  134.   
  135.   
  136. 五、 基本字符流类  
  137. 图阐述了Reader和Writer字符流的体系结构。  
  138. 5.一、InputStreamReader 和 OutputStreamWriter  
  139. 用于字节流与字符流之间的转换接口。  
  140. 当你构造一个InputStreamReader或OutputStreamWriter时,转换规则定义了16位Unicode和其它平台的特定表示之间的转换。  
  141. InputStreamReader从一个数据源读取字节,并自动将其转换成Unicode字符。  
  142. 若是你特别声明,InputStreamReade会将字节流转换成其它种类的字符流。  
  143. OutputStreamWriter将字符的Unicode编码写到输出流,若是你的使用的不是Unicode字符,OutputStreamWriter会将你的字符编码转换成Unicode编码。  
  144. 5.2.、缓冲读者和做者  
  145. 由于在各类格式之间进行转换和其它I/O操做很相似,因此在处理大块数据时效率最高。  
  146. 在InputStreamReader和OutputStreamWriter的结尾连接一个BufferedReader和BufferedWriter是一个好主意。  
  147. 记住对BufferedWriter使用flush()方法。  
  148. 5.三、 使用其它字符转换  
  149. 若是你须要从一个非本地(例如,从链接到一个不一样类型的机器的网络链接读取)的字符编码读取输入,  
  150. 你能够象下面这个程序那样,使用显式的字符编码构造ir=new InputStreamReader(System.in,  “8859_1″);  
  151. 注:若是你经过网络链接读取字符,就应该使用这种形式。  
  152. 不然,你的程序会老是试图将所读取的字符看成本地表示来进行转换,而这并不老是正确的。ISO 8859-1是映射到ASCII的Latin-1编码模式。  
  153.   
  154.   
  155. 六、 对象串行化  
  156. java.io.Serializable接口支持将一个Java技术对象存放到一个流中。  
  157. 将一个对象存放到某种类型的永久存储器上称为”保持”。  
  158. 若是一个对象能够被存放到磁盘或磁带上,或者能够发送到另一台机器并存放到存储器或磁盘上,那么这个对象就被称为可保持的。  
  159. java.io.Serializable接口没有任何方法,它只做为一个”标记”,用来代表实现了这个接口的类能够串行化。  
  160. 类中没有实现Serializable接口的对象不能被保持。  
  161. // 文件实现追加:  
  162. // 其中的FileWriter()中的第二个参数的含义是:是否在文件中追加内容  
  163. PrintWriter out = new PrintWriter(new FileWriter(logFileName, true), true);  
  164. Java读写文件最经常使用的类是FileInputStream/FileOutputStream和FileReader/FileWriter。  
  165. 其中FileInputStream和FileOutputStream是基于字节流的,经常使用于读写二进制文件。  
  166. 读写字符文件建议使用基于字符的FileReader和FileWriter,省去了字节与字符之间的转换。  
  167. 但这两个类的构造函数默认使用系统的编码方式,若是文件内容与系统编码方式不一致,可能会出现乱码。  
  168. 在这种状况下,建议使用FileReader和FileWriter的父类:InputStreamReader/OutputStreamWriter,  
  169. 它们也是基于字符的,但在构造函数中能够指定编码类型:InputStreamReader(InputStream in, Charset cs) 和OutputStreamWriter(OutputStream out, Charset cs)。    
  170.   
  171.    
  172.   
  173.    
  174.   
  175. // 读写文件的编码:  
  176. InputStreamReader r = new InputStreamReader(new FileInputStream(fileName), “utf-8″);  
  177. OutputStreamWriter out = new OutputStreamWriter(new FileOutputStream(fileName),”utf-8″);  
  178.   
  179. /** 
  180. 三种IO性能比较: 
  181. 在读写一个10k文件的时候,三种方式的耗时以下: 
  182. InputStreamReader And OutputStreamWriter : 63ms (能够设置文件的编码,若是不用buffer) 
  183. BufferedReader And BufferedWriter : 31ms  
  184. BufferedInputStream And BufferedOutputStream : 16ms 
  185. */  
  186.   
  187. /** 
  188. * Description: Test the java IO’s efficiency 
  189. * Author: AllanCao 
  190. * Date: 2007-02-18 
  191. */  
  192. import java.io.*;  
  193.   
  194. /** 
  195. * using the InputStreamReader And OutputStreamWriter 
  196. */  
  197. class EncoderRW {  
  198.  public static String read(String fileName) throws IOException {  
  199.   StringBuffer sb = new StringBuffer();  
  200.   /*此处读文件时用了buffer,若是不用,性能损失一倍*/  
  201.   BufferedReader in = new BufferedReader(new InputStreamReader(new FileInputStream(fileName), “utf-8″));  
  202.   String s;  
  203.   while((s = in.readLine()) != null) {  
  204.     sb.append(s);  
  205.     sb.append(”\n”);  
  206.   }  
  207.   in.close();  
  208.   return sb.toString();  
  209.  }  
  210.  public void write(String fileName, String text) throws IOException {  
  211.   OutputStreamWriter out = new OutputStreamWriter(new FileOutputStream(fileName),”utf-8″);  
  212.   out.write(text);  
  213.   out.flush();  
  214.   out.close();  
  215.  }  
  216. }  
  217.   
  218. /** 
  219. * using the BufferedReader And BufferedWriter 
  220. */  
  221. class WriterReader {  
  222.  public String read(String fileName) throws IOException {  
  223.   StringBuffer sb = new StringBuffer();  
  224.   BufferedReader in = new BufferedReader(new FileReader(fileName));  
  225.   String s;  
  226.   while((s = in.readLine()) != null) {  
  227.     sb.append(s);  
  228.     sb.append(”\n”);  
  229.   }  
  230.   in.close();  
  231.   return sb.toString();  
  232.  }  
  233.  public void write(String fileName, String text) throws IOException {  
  234.   PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(fileName)));  
  235.   out.print(text);  
  236.   out.close();  
  237.  }  
  238. }  
  239.   
  240. /** 
  241. * using the BufferedInputStream And BufferedOutputStream 
  242. */  
  243. class BufferedStream{  
  244.  public byte[] read(String fileName) throws IOException {  
  245.   BufferedInputStream remoteBIS = new BufferedInputStream(new FileInputStream(fileName));  
  246.   ByteArrayOutputStream baos = new ByteArrayOutputStream(10240);  
  247.   byte[] buf = new byte[1024];  
  248.   int bytesRead = 0;  
  249.   while(bytesRead >= 0)  
  250.   {  
  251.    baos.write(buf, 0, bytesRead);  
  252.    bytesRead = remoteBIS.read(buf);  
  253.   }  
  254.   byte[] content = baos.toByteArray();  
  255.   return content;  
  256.  }  
  257.  public void write(String fileName, byte[] content)  throws IOException {  
  258.   BufferedOutputStream out = new BufferedOutputStream(new FileOutputStream(fileName));  
  259.   out.write(content);  
  260.   out.flush();  
  261.   out.close();  
  262.  }  
  263. }  
  264.   
  265. public class TestIO  
  266. {  
  267.  public static void main(String[] args)throws IOException {  
  268.   long currentTime = System.currentTimeMillis() ;  
  269.   EncoderRW rw = new EncoderRW();  
  270.   rw.write(”index.dat”,rw.read(”FileUtil.java”));  
  271.   System.out.println(”cost time:” + Long.toString(System.currentTimeMillis()-currentTime) + ” ms”);  
  272.   
  273.   currentTime = System.currentTimeMillis() ;  
  274.   WriterReader wr = new WriterReader();  
  275.   wr.write(”index.dat”,wr.read(”FileUtil.java”));  
  276.   System.out.println(”cost time:” + Long.toString(System.currentTimeMillis()-currentTime) + ” ms”);  
  277.   
  278.   currentTime = System.currentTimeMillis() ;  
  279.   BufferedStream bf = new BufferedStream();  
  280.   bf.write(”index.dat”,bf.read(”FileUtil.java”));  
  281.   System.out.println(”cost time:” + Long.toString(System.currentTimeMillis()-currentTime) + ” ms”);  
  282.  }  
  283. }  
  284. http://dinghaoliang.blog.163.com/blog/static/12654071420106921425505/