JAVASE IO流

前言

本文以图文并茂的形式重点记录了这一周学习Java中IO操作的心得,并配以大量练习代码。Java的IO流无外乎就是输入流和输出流,所以基础部分还是比较简单的。

简述

IO:用于处理设备上的数据的技术。设备:内存、硬盘、光盘。java中所涉及的功能对象都存储到java.io包中。

:系统资源,windows系统本身就可以操作设备。各种语言只是使用了系统平台上的这个资源。并对外提供了各种语言自己的操作功能,这些功能最终调用的是系统资源。使用完资源一定要记住:释放。(也就是说IO一定要写finally!)

IO流也进行分类:

1:输入流(读)和输出流(写)。
2:因为处理的数据不同,分为字节流和字符流。


输入流和输出流相对于内存设备而言。
将外设中的数据读取到内存中:输入-->读
将内存的数写入到外设中:输出-->写
注意:流的操作只有两种:读和写。


字节流:处理字节数据的流对象。设备上的数据无论是图片或者dvd,文字,它们都以二进制存储的。二进制的最终都是以一个8位为数据单元进行体现,所以计算机中的最小数据单元就是字节。意味着,字节流可以处理设备上的所有数据,所以字节流一样可以处理字符数据。

字符流:因为字符每个国家都不一样,所以涉及到了字符编码问题,那么GBK编码的中文用unicode编码解析是有问题的,所以需要获取中文字节数据的同时+ 指定的编码表才可以解析正确数据。为了方便于文字的解析,所以将字节流和编码表封装成对象,这个对象就是字符流。只要操作字符数据,优先考虑使用字符流体系。

流的体系因为功能不同,但是有共性内容,不断抽取,形成继承体系。该体系一共有四个基类,而且都是抽象类。
字节流:InputStream  OutputStream
字符流:Reader  Writer

在这四个系统中,它们的子类,都有一个共性特点:子类名后缀都是父类名,前缀名都是这个子类的功能名称。

File类

java.io.File类将文件系统中的文件和文件夹封装成了对象。提供了更多的属性和行为可以对这些文件和文件夹进行操作。这些是流对象办不到的,因为流只操作数据。设备上的数据(0、1串)最常见的存储表现形式是文件file。先学习一下文件的基本操作。
查阅API,描述文件或者文件夹(目录路径名)的类是File类。常用方法需要查表。

首先我们来演示一下File构造函数的使用。

[java]  view plain  copy
  1. package ustc.lichunchun.file.demo;  
  2.   
  3. import java.io.File;  
  4. /* 
  5.  * File对象创建过程、字段信息。 
  6.  *  
  7.  * 注意:File文件对象只能操作文件或者文件夹的属性, 
  8.  *       例如文件或文件夹的创建、删除、获取文件属性(大小、所在目录等), 
  9.  *       但我们最终建立文件的目的,是往文件里面存数据,File对象是做不了这个的。 
  10.  * 这时,我们就要用到IO流。 
  11.  */  
  12. public class FileDemo {  
  13.   
  14.     //File类已经提供了相应字段。  
  15.     //private static final String FILE_SEPARATOR = System.getProperty("file.separator");  
  16.   
  17.     public static void main(String[] args) {  
  18.           
  19.         //将某一个文件或者文件夹封装成了File对象。可以封装存在的文件或目录,也可以封装不存在的文件或目录。  
  20.         //注意,这里File对象封装的实际上是1.txt,至于d:\\只是作为全路径目录。  
  21.         //再比如,new File("abc\\a\\b\\c");实际上封装的是c文件夹以及其全路径目录。  
  22.         File file = new File("d:\\1.txt");  
  23.           
  24.         //File(String parent, String child);这样可以将目录和文件名分开。  
  25.         File file1 = new File("d:\\", "1.txt");  
  26.           
  27.         File dir = new File("d:\\");  
  28.         File file2 = new File(dir, "1.txt");  
  29.           
  30.         //File f = new File("d:"+System.getProperty("file.separator")+"abc"+System.getProperty("file.separator")+"1.txt");  
  31.           
  32.         //File f = new File("d:"+FILE_SEPARATOR+"abc"+FILE_SEPARATOR+"1.txt");  
  33.           
  34.         File f = new File("d:"+File.separator+"abc"+File.separator+"1.txt");  
  35.           
  36.         System.out.println(f);  
  37.     }  
  38. }  
如果把某个文件视为一个对象,让你来描述这个文件,你认为,它应该具备什么样的功能或者方法?

File类常见方法
1:创建。
boolean createNewFile():在指定目录下创建文件,如果该文件已存在,则不创建。而对操作文件的输出流而言,比如FileOutputStream,输出流对象已建立,就会创建文件,如果文件已存在,会覆盖。除非续写。
boolean mkdir():创建此抽象路径名指定的目录。
boolean mkdirs():创建多级目录。 
2:删除。
boolean delete():删除此抽象路径名表示的文件或目录。
void deleteOnExit():在虚拟机退出时删除。
注意:在删除文件夹时,必须保证这个文件夹中没有任何内容,才可以将该文件夹用delete删除。
window的删除动作,是从里往外删。注意:java删除文件不走回收站。要慎用。
3:获取。
long length():获取文件大小。
String getName():返回由此抽象路径名表示的文件或目录的名称。    
String getPath():将此抽象路径名转换为一个路径名字符串。
String getAbsolutePath():返回此抽象路径名的绝对路径名字符串。
String getParent():返回此抽象路径名父目录的抽象路径名,如果此路径名没有指定父目录,则返回 null。
long lastModified():返回此抽象路径名表示的文件最后一次被修改的时间。
File.pathSeparator:返回当前系统默认的路径分隔符,windows默认为 “;”。
File.Separator:返回当前系统默认的目录分隔符,windows默认为 “\”。
4:判断。
boolean exists():判断文件或者文件夹是否存在。
boolean isDirectory():测试此抽象路径名表示的文件是否是一个目录。
boolean isFile():测试此抽象路径名表示的文件是否是一个标准文件。
boolean isHidden():测试此抽象路径名指定的文件是否是一个隐藏文件。
boolean isAbsolute():测试此抽象路径名是否为绝对路径名。
5:重命名。
boolean renameTo(File dest):可以实现移动的效果。剪切+重命名。

String[] list():列出指定目录下的当前的文件和文件夹的名称。包含隐藏文件。如果调用list方法的File 对象中封装的是一个文件,那么list方法返回数组为null。如果封装的对象不存在也会返回null。只有封装的对象存在并且是文件夹时,这个方法才有效。

注意:list(FilenameFilter filter)和listFiles(FileFilter filter)往往是和过滤器实例参数一起使用。

代码示例1:

[java]  view plain  copy
  1. package ustc.lichunchun.file.demo;  
  2.   
  3. import java.io.File;  
  4.   
  5. public class FileMethodDemo {  
  6.   
  7.     public static void main(String[] args) {  
  8.   
  9.         /* 
  10.          * File类,常见方法。 
  11.          * 1.名字。获取名称。 
  12.          *      String getName(); 
  13.          * 2.大小。获取大小。 
  14.          *      long length(); 
  15.          * 3.类型。获取类型。 
  16.          *      没有,因为类型可以自定义。 
  17.          * 4.获取所在目录。 
  18.          *      String getParent(); 
  19.          */  
  20.           
  21.         File file = new File("d:\\abc\\1.txt");  
  22.         String file_name = file.getName();  
  23.         System.out.println(file_name);//1.txt-->File对象封装的是1.txt所在路径(而且不是绝对路径)。  
  24.           
  25.         long len = file.length();  
  26.         System.out.println(len);//0  
  27.           
  28.         System.out.println(file.getParent());//d:\abc  
  29.     }  
  30. }  
代码示例2:
[java]  view plain  copy
  1. package ustc.lichunchun.file.demo;  
  2.   
  3. import java.io.File;  
  4. import java.io.IOException;  
  5. import java.text.DateFormat;  
  6. import java.util.Date;  
  7.   
  8. public class FileMethodTest {  
  9.   
  10.     public static void main(String[] args) throws IOException {  
  11.         /* 
  12.          * File方法 练习: 
  13.          *  
  14.          * 1.获取文件的绝对路径。 
  15.          *  String getAbsolutePath(); 
  16.          *  
  17.          * 2.获取文件的路径。 
  18.          *  String getPath(); 
  19.          *  
  20.          * 3.获取文件最后一次修改的时间。要求是x年x月x日。时间。 
  21.          *  long lastModified(); 
  22.          *  
  23.          * 4.文件是否是隐藏的。 
  24.          *  boolean isHidden(); 
  25.          *  
  26.          * 5.发现File对象封装的文件或者文件夹是可以存在的也可以是不存在的。 
  27.          * 那么不存在的可否用file的功能创建呢? 
  28.          * 创建功能。 
  29.          *  boolean createNewFile(); 
  30.          *  
  31.          * 删除功能。 
  32.          *  boolean delete(); 
  33.          *  
  34.          * 6.一个File对象封装成的文件或者文件夹到底是否存在呢? 
  35.          * 判断存在功能。 
  36.          *  boolean exists(); 
  37.          *  
  38.          * 7.getFreeSpace()方法是什么意思?用Demo验证。getTotalSpace()、getUsableSpace() 
  39.          *  指定分区中:未分配、总共、已分配字节数。用处:迅雷看看缓存到本地,会先判断哪个盘符剩余空间比较大。 
  40.          *  
  41.          * 8.列出可用的文件系统根。 
  42.          *  file[] listRoots(); 
  43.          *  
  44.          */  
  45.         //methodDemo1();  
  46.           
  47.         //文件创建与删除  
  48.         File file = new File("1.txt");  
  49.         //methodDemo2(file);  
  50.   
  51.         //文件夹创建与删除  
  52.         File file1 = new File("abc\\a\\b\\c");  
  53.         //methodDemo3(file1);  
  54.           
  55.         //揭秘  
  56.         //methodDemo4();  
  57.           
  58.         File file2 = new File("d:\\");  
  59.         //System.out.println(file2.getFreeSpace());//128448581632  
  60.           
  61.         listRootsDemo();  
  62.           
  63.     }  
  64.   
  65.     public static void methodDemo1() {  
  66.         File file = new File("abc\\1.txt");  
  67.         String path = file.getAbsolutePath();//E:\JavaSE_code\day21e\abc\1.txt-->获取文件对象的绝对路径。即使封装的是相对的,获取到的也是绝对的,这时就是所在项目的绝对路径下。  
  68.         String path1 = file.getPath();//abc\1.txt-->获取的是file对象中的封装的路径。封装的是什么,获取到的就是什么。  
  69.         System.out.println("AbsolutePath = "+path);  
  70.         System.out.println("Path = "+path1);  
  71.   
  72.         File file1 = new File("E:\\JavaSE_code\\day21e\\IO流_1.txt");  
  73.         long time = file1.lastModified();  
  74.         Date date = new Date(time);  
  75.         String str_date = DateFormat.getDateTimeInstance(DateFormat.LONG,DateFormat.LONG).format(date);  
  76.         System.out.println(time);//1436701878494  
  77.         System.out.println(str_date);//2015年7月12日 下午07时51分18秒  
  78.           
  79.         boolean b = file.isHidden();  
  80.         System.out.println("isHidden():" + b);//false  
  81.     }  
  82.       
  83.     public static void methodDemo2(File file) throws IOException {  
  84.           
  85.         //1.演示文件的创建。  
  86.         boolean b = file.createNewFile();//如果文件存在,则不创建,返回false;不存在,就创建,创建成功,返回true。  
  87.         System.out.println(b);  
  88.           
  89.         //2.文件的删除。  
  90.         boolean b1 = file.delete();  
  91.         System.out.println(b1);  
  92.           
  93.         //3.判断文件是否存在  
  94.         System.out.println(file.exists());  
  95.     }  
  96.       
  97.     public static void methodDemo3(File file) {  
  98.           
  99.         //1.演示文件夹的创建。  
  100.         //boolean b = file.mkdir();//只能创建单级目录:new File("abc")  
  101.         boolean b = file.mkdirs();//创建多级目录:new File("abc\\a\\b\\c"),或者创建单级目录:new File("abc")  
  102.         System.out.println(b);  
  103.         System.out.println("exists: "+file.exists());  
  104.           
  105.         //2.演示文件夹的删除。  
  106.         boolean b1 = file.delete();//删除文件夹时,必须保证该文件夹没有内容。有内容,必须先把内容删除后,才可以删除当前文件夹。  
  107.         System.out.println("delete: "+b1);  
  108.     }  
  109.       
  110.     public static void methodDemo4() throws IOException {  
  111.         //揭秘:你用什么方法创建的,它就是什么。  
  112.         //千万不要被表面现象所迷惑: abc.txt有可能是文件夹,abc有可能是文件。  
  113.         //是不是文件,不能主观判断,得用isFile()、isDirectory()判断。  
  114.         //而且在用is之前,要先判断是否存在!  
  115.         File file1 = new File("abc");  
  116.         System.out.println("file: "+file1.isFile());//false  
  117.         System.out.println("directory: "+file1.isDirectory());//false,如果一个东西不存在的情况下,不可能是文件或者文件夹。  
  118.         System.out.println("--------------");  
  119.         File file2 = new File("abc");  
  120.         boolean b2 = file2.mkdirs();  
  121.         System.out.println("mkdirs: "+b2);//true  
  122.         System.out.println("file: "+file2.isFile());//false  
  123.         System.out.println("directory: "+file2.isDirectory());//true  
  124.         System.out.println("--------------");  
  125.         File file3 = new File("edf");  
  126.         boolean b3 = file3.createNewFile();  
  127.         System.out.println("createNewFile: "+b3);//true  
  128.         System.out.println("file: "+file3.isFile());//true,文件不一定非要有扩展名  
  129.         System.out.println("directory: "+file3.isDirectory());//false  
  130.         System.out.println("--------------");  
  131.         File file4 = new File("abc.txt");  
  132.         boolean b4 = file4.mkdirs();  
  133.         System.out.println("mkdirs: "+b4);//true  
  134.         System.out.println("file: "+file4.isFile());//false  
  135.         System.out.println("directory: "+file4.isDirectory());//true,文件夹也可以包含"."  
  136.     }  
  137.       
  138.     public static void listRootsDemo() {  
  139.         File[] files = File.listRoots();  
  140.         for(File file : files)  
  141.             System.out.println(file);  
  142.     }  
  143. }  
代码示例3:
[java]  view plain  copy
  1. package ustc.lichunchun.file.demo;  
  2.   
  3. import java.io.File;  
  4.   
  5. public class FileMethodTest2 {  
  6.   
  7.     public static void main(String[] args) {  
  8.   
  9.         /* 
  10.          * 9.获取指定文件夹中的所有文件和文件夹的名称。 
  11.          */  
  12.           
  13.         File dir = new File("d:\\");  
  14.           
  15.         String[] names = dir.list();//列出当前目录下的所有文件和文件夹名称,包含隐藏文件。  
  16.                                     //list()局限:只获取名称。  
  17.                                     //如果目录存在但是没有内容,会返回一个数组,但是长度为0。  
  18.         if(names != null){  
  19.             for (String name : names) {  
  20.                 System.out.println(name);  
  21.             }  
  22.         }  
  23.         System.out.println("-------------------------");  
  24.           
  25.         File[] files = dir.listFiles();//获取当前目录下的所有文件和文件夹的File对象,更为常用。  
  26.         for(File f : files){  
  27.             System.out.println(f.getName()+"......"+f.length());  
  28.         }  
  29.     }  
  30. }  
过滤器

接下来我要介绍的是文件名过滤器和文件过滤器,它们的用途是为了获取指定目录下的指定类型文件。所采用到的设计模式属于策略设计模式,目的就是为了降低容器和过滤条件之间的耦合性。

1.文件名过滤器FilenameFilter

它的底层源码如下:

[java]  view plain  copy
  1. public String[] list(FilenameFilter filter) {  
  2.     String names[] = list();  
  3.     if ((names == null) || (filter == null)) {  
  4.         return names;  
  5.     }  
  6.     List v = new ArrayList<>();  
  7.     for (int i = 0 ; i < names.length ; i++) {  
  8.         if (filter.accept(this, names[i])) {  
  9.             v.add(names[i]);  
  10.         }  
  11.     }  
  12.     return v.toArray(new String[v.size()]);  
  13. }  
下面实现一个判断文件是否是以.java格式结尾的过滤器:
[java]  view plain  copy
  1. package ustc.lichunchun.filter;  
  2.   
  3. import java.io.File;  
  4. import java.io.FilenameFilter;  
  5. /* 
  6.  * 根据文件名称的后缀名进行过滤的过滤器。 
  7.  */  
  8. public class FilterBySuffix implements FilenameFilter {  
  9.   
  10.     private String suffix;  
  11.       
  12.     public FilterBySuffix(String suffix) {  
  13.         super();  
  14.         this.suffix = suffix;  
  15.     }  
  16.   
  17.     /** 
  18.      * @param name 被遍历目录dir中的文件夹或者文件的名称。 
  19.      */  
  20.     @Override  
  21.     public boolean accept(File dir, String name) {  
  22.           
  23.         return name.endsWith(suffix);  
  24.     }  
  25. }  
示例1:获取指定目录下的.java文件:
[java]  view plain  copy
  1. package ustc.lichunchun.file.demo;  
  2.   
  3. import java.io.File;  
  4.   
  5. import ustc.lichunchun.filter.FilterBySuffix;  
  6.   
  7. public class FilenameFilterDemo {  
  8.   
  9.     public static void main(String[] args) {  
  10.   
  11.         /* 
  12.          * 10.能不能只获取指定目录下的.java文件呢? 
  13.          *  文件名过滤器:list(FilenameFilter filter); 
  14.          */  
  15.           
  16.         /* 
  17.         File dir = new File("d:\\"); 
  18.         String[] names = dir.list(); 
  19.         for(String name : names){ 
  20.             if(name.endsWith(".java"))//-->耦合性太强。 
  21.                 System.out.println(name); 
  22.         } 
  23.         */  
  24.           
  25.         //文件名过滤器:让容器和过滤条件分离,降低耦合性。  
  26.         //类似于比较器,都属于策略设计模式。不要面对具体的过滤或者排序动作,我只面对接口。  
  27.         File dir = new File("d:\\");  
  28.         //传入一个过滤器。  
  29.         String[] names = dir.list(new FilterBySuffix(".java"));  
  30.         for(String name : names){  
  31.             System.out.println(name);  
  32.         }  
  33.     }  
  34. }  
我又实现了一个文件名包含指定字段的过滤器:
[java]  view plain  copy
  1. package ustc.lichunchun.filter;  
  2.   
  3. import java.io.File;  
  4. import java.io.FilenameFilter;  
  5.   
  6. public class FilterByContain implements FilenameFilter {  
  7.   
  8.     private String content;  
  9.       
  10.     public FilterByContain(String content) {  
  11.         super();  
  12.         this.content = content;  
  13.     }  
  14.   
  15.     @Override  
  16.     public boolean accept(File dir, String name) {  
  17.   
  18.         return name.contains(content);  
  19.     }  
  20. }  
示例2:获取指定目录下,文件名中包含指定字段的文件。

[java]  view plain  copy
  1. package ustc.lichunchun.file.demo;  
  2.   
  3. import java.io.File;  
  4. import java.io.FilenameFilter;  
  5.   
  6. import ustc.lichunchun.filter.FilterByContain;  
  7. import ustc.lichunchun.filter.FilterBySuffix;  
  8.   
  9. public class FilenameFilterDemo2 {  
  10.   
  11.     public static void main(String[] args) {  
  12.   
  13.         //需求:不是获取指定后缀名的文件,而是获取文件名中包含指定字段的文件。  
  14.         File dir = new File("d:\\");  
  15.           
  16.         FilenameFilter filter = new FilterBySuffix(".java");//过滤后缀名的过滤器。  
  17.         filter = new FilterByContain("Demo");//过滤内容的过滤器。  
  18.           
  19.         String[] names = dir.list(filter);  
  20.           
  21.         for(String name : names){  
  22.             System.out.println(name);  
  23.         }  
  24.     }  
  25. }  

2.文件过滤器FileFilter

文件过滤器其实更为常用。因为过滤器中pathname.getName().endsWith(".java")可以实现同样的文件名过滤操作。

我再用文件过滤器的方法,实现上面文件名过滤器所示例的,过滤指定类型文件的过滤器:

[java]  view plain  copy
  1. package ustc.lichunchun.filter;  
  2.   
  3. import java.io.File;  
  4. import java.io.FileFilter;  
  5.   
  6. public class FilterBySuffix2 implements FileFilter {  
  7.   
  8.     private String suffix;  
  9.       
  10.     public FilterBySuffix2(String suffix) {  
  11.         super();  
  12.         this.suffix = suffix;  
  13.     }  
  14.   
  15.     @Override  
  16.     public boolean accept(File pathname) {  
  17.   
  18.         return pathname.getName().endsWith(suffix);  
  19.     }  
  20. }  
为了更好地让读者理解,下面还实现了当前目录下,只过滤出文件和只过滤出文件夹的文件过滤器:
[java]  view plain  copy
  1. package ustc.lichunchun.filter;  
  2.   
  3. import java.io.File;  
  4. import java.io.FileFilter;  
  5.   
  6. public class FilterByFile implements FileFilter {  
  7.   
  8.     @Override  
  9.     public boolean accept(File pathname) {  
  10.   
  11.         return pathname.isFile();//文件过滤器。只筛选出文件,不要文件夹。  
  12.     }  
  13. }  
[java]  view plain  copy
  1. package ustc.lichunchun.filter;  
  2.   
  3. import java.io.File;  
  4. import java.io.FileFilter;  
  5.   
  6. public class FilterByDirectory implements FileFilter {  
  7.   
  8.     @Override  
  9.     public boolean accept(File pathname) {  
  10.   
  11.         return pathname.isDirectory();  
  12.     }  
  13. }  
示例3:
[java]  view plain  copy
  1. package ustc.lichunchun.file.demo;  
  2.   
  3. import java.io.File;  
  4. import java.io.FileFilter;  
  5.   
  6. import ustc.lichunchun.filter.FilterByDirectory;  
  7. import ustc.lichunchun.filter.FilterByFile;  
  8. import ustc.lichunchun.filter.FilterBySuffix2;  
  9.   
  10. public class FileFilterDemo {  
  11.   
  12.     public static void main(String[] args) {  
  13.   
  14.         File dir = new File("d:\\");  
  15.           
  16.         FileFilter filter = new FilterByFile();//过滤出当前目录下所有文件  
  17.         filter = new FilterByDirectory();//过滤出当前目录下所有文件夹  
  18.         filter = new FilterBySuffix2(".java");//过滤出当前目录下所有以指定后缀名结尾的文件和文件夹  
  19.           
  20.         File[] files = dir.listFiles(filter);  
  21.           
  22.         for(File file : files){  
  23.             System.out.println(file);  
  24.         }  
  25.         System.out.println("-------------------------");  
  26.     }  
  27. }  
练习:编写一个完整的Java Application程序。主类为NumberArray,要求NumberArray能够通过对接口NumberFilter的调用完成从一组整数中过滤出满足NumberArray使用者筛选条件的部分整数,设计中涉及到的类和接口的具体要求和含义如下:

(1)接口NumberFilter:表示过滤器接口。接口NumberFilter的抽象方法boolean is(int n)供使用者实现以设置筛选条件。

(2)类NumberArray:表示一组整数的对象(可用int型数组作为其成员来存放一组整数)。类NumberArray的方法print(0输出满足过滤器NumberFilter设置的筛选条件的部分整数。类NumberArray的方法setFilter(NumberFilter nf)用于设置过滤器。

(3)类Filter:表示具体的过滤器,实现接口NumberFilter,实现is()方法用于设置筛选条件,这里要求过滤出偶数。

[java]  view plain  copy
  1. package ustc.lichunchun.filter;  
  2. public class Test {  
  3.     public static void main(String[] args) {  
  4.         int[] array = {1,2,3,4,5,6,7,8,9};  
  5.         NumberFilter nf = new Filter();  
  6.         NumberArray na = new NumberArray(array, nf);  
  7.         na.print();  
  8.     }  
  9. }  
  10. class NumberArray {  
  11.     private int[] array;  
  12.     public NumberArray(int[] array, NumberFilter nf) {  
  13.         super();  
  14.         this.array = array;  
  15.         this.nf = nf;  
  16.     }  
  17.     private NumberFilter nf;  
  18.     public void print(){  
  19.         for(int i : array){  
  20.             if(nf.is(i)){  
  21.                 System.out.print(i+ " ");  
  22.             }  
  23.         }  
  24.         System.out.println();  
  25.     }  
  26.     public void setFilter(NumberFilter nf){  
  27.         this.nf = nf;  
  28.     }  
  29. }  
  30. class Filter implements NumberFilter {  
  31.     @Override  
  32.     public boolean is(int n) {  
  33.         if(n % 2 == 0)  
  34.             return true;  
  35.         return false;  
  36.     }  
  37. }  
  38. interface NumberFilter {  
  39.     boolean is(int n);  
  40. }  

递归及其IO应用

说完了上面的过滤器以后,我们已经可以实现过滤出指定目录下的特定类型文件了。但是现在我又有一个需求:遍历指定目录下的内容(包含子目录中的内容)。这该如何处理呢?再解决这个问题之前,我们下来简单回顾一下算法课中所学的递归思想

递归:就是函数自身调用自身。
什么时候用递归呢?当一个功能被重复使用,而每一次使用该功能时的参数不确定,都由上次的功能元素结果来确定。简单说:功能内部又用到该功能,但是传递的参数值不确定。(每次功能参与运算的未知内容不确定)。
递归的注意事项:
1:一定要定义递归的条件。
2:递归的次数不要过多。容易出现 StackOverflowError 栈内存溢出错误。
其实递归就是在栈内存中不断的加载同一个函数。

递归示例:

[java]  view plain  copy
  1. package ustc.lichunchun.recursion;  
  2.   
  3. public class RecursionDemo {  
  4.   
  5.     public static void main(String[] args) {  
  6.   
  7.         /* 
  8.          * 递归使用时,一定要定义条件。 
  9.          * 注意:递归次数过多,会出现栈内存溢出。 
  10.          */  
  11.         //show();  
  12.           
  13.         int sum = getSum(3);  
  14.         System.out.println("sum = "+sum);//3+((3-1)+(3-1-1))  
  15.           
  16.         int sum1 = getSum(999999);//java.lang.StackOverflowError  
  17.     }  
  18.       
  19.     public static int getSum(int num){  
  20.         if(num == 1)  
  21.             return 1;  
  22.         return num + getSum(num - 1);  
  23.     }  
  24.       
  25.     /*这也是递归,并且会溢出。 
  26.     public static void show(){ 
  27.         method(); 
  28.     } 
  29.     public static void method(){ 
  30.         show(); 
  31.     } 
  32.     */  
  33. }  
IO递归示例1:遍历指定目录下的内容,要求包含子目录的内容
[java]  view plain  copy
  1. package ustc.lichunchun.file.test;  
  2.   
  3. import java.io.File;  
  4.   
  5. public class GetAllFilesTest {  
  6.   
  7.     public static void main(String[] args) {  
  8.           
  9.         /* 
  10.          * 遍历指定目录下的内容(包含子目录中的内容) 
  11.          *  
  12.          * 递归:函数自身调用自身,不断进栈。函数内部又使用到了该函数功能。 
  13.          * 什么时候使用呢? 
  14.          * 功能被重复使用,但是每次该功能使用参与运算的数据不同时,可以考虑递归方式解决。 
  15.          *  
  16.          */  
  17.         File dir = new File("d:\\JavaSE_code");  
  18.         getAllFiles(dir);  
  19.           
  20.     }  
  21.     public static void getAllFiles(File dir){  
  22.           
  23.         System.out.println("dir: "+dir);  
  24.           
  25.         //1.获取该目录的文件对象数组  
  26.         File[] files = dir.listFiles();  
  27.           
  28.         //2.对数组进行遍历  
  29.         if(files != null){//windows一些文件夹是不可以被java访问到的。  
  30.             for(File file : files){  
  31.                 if(file.isDirectory()){  
  32.                     getAllFiles(file);  
  33.                 }else{  
  34.                     System.out.println("file: "+file);  
  35.                 }  
  36.             }  
  37.         }  
  38.     }  
  39. }  
IO递归示例2:删除一个带内容的文件夹
[java]  view plain  copy
  1. package ustc.lichunchun.file.test;  
  2.   
  3. import java.io.File;  
  4.   
  5. public class DeleteDirTest {  
  6.   
  7.     public static void main(String[] args) {  
  8.   
  9.         /* 
  10.          * 基于递归,做一个练习:删除一个带内容的文件夹。必须从里往外删。 
  11.          */  
  12.           
  13.         File dir = new File("d:\\JavaSE_code");  
  14.           
  15.         //System.out.println(dir.delete());//false,有内容的文件夹不能直接删  
  16.           
  17.         deleteDir(dir);  
  18.     }  
  19.     public static void deleteDir(File dir){  
  20.           
  21.         //1.列出当前目录下的文件以及文件夹。  
  22.         File[] files = dir.listFiles();  
  23.           
  24.         //2.对该数组进行遍历。  
  25.         for(File file : files){  
  26.               
  27.             //3/判断是否有目录。如果有,继续使用该功能遍历,递归!如果不是文件夹,直接删除。  
  28.             if(file.isDirectory()){  
  29.                 deleteDir(file);  
  30.             }else{  
  31.                 System.out.println(file + ":" +file.delete());  
  32.             }  
  33.         }  
  34.         //4.删除不含文件了的文件夹  
  35.         System.out.println(dir + ":" +dir.delete());  
  36.     }  
  37. }  

File类综合练习

获取一个想要的指定文件的集合。获取JavaSE_code下(包含子目录)的所有的.java的文件对象。并存储到集合中。

思路:
1.既然包含子目录,就需要递归。
2.在递归的过程中,需要过滤器。
3.凡是满足条件的,都添加到集合中。

[java]  view plain  copy
  1. package ustc.lichunchun.test;  
  2.   
  3. import java.io.File;  
  4. import java.io.FileFilter;  
  5. import java.util.ArrayList;  
  6. import java.util.List;  
  7.   
  8. import ustc.lichunchun.filter.FilterBySuffix2;  
  9.   
  10. public class Test {  
  11.   
  12.     public static void main(String[] args) {  
  13.   
  14.         /* 
  15.          * 需求:获取一个想要的指定文件的集合。获取JavaSE_code下(包含子目录)的所有的.java的文件对象。并存储到集合中。 
  16.          *  
  17.          * 思路: 
  18.          * 1.既然包含子目录,就需要递归。 
  19.          * 2.在递归的过程中,需要过滤器。 
  20.          * 3.凡是满足条件的,都添加到集合中。 
  21.          */  
  22.         File dir = new File("e:\\JavaSE_code");  
  23.         List list = fileList(dir, ".java");  
  24.         for(File file : list){  
  25.             System.out.println(file);  
  26.         }  
  27.     }  
  28.     /** 
  29.      * 定义一个获取指定过滤器条件的文件的集合。 
  30.      */  
  31.     public static List fileList(File dir, String suffix){  
  32.           
  33.         //1.定义集合  
  34.         List list = new ArrayList();  
  35.           
  36.         //2.定义过滤器。  
  37.         FileFilter filter = new FilterBySuffix2(suffix);  
  38.         /*匿名内部类也可以,不过不建议这么做。 
  39.         FileFilter filter = new FileFilter(){ 
  40.  
  41.             @Override 
  42.             public boolean accept(File pathname) { 
  43.                  
  44.                 return pathname.getName().endsWith(suffix); 
  45.             } 
  46.         };*/  
  47.         getFileList(dir, list, filter);  
  48.         return list;  
  49.     }  
  50.       
  51.     /** 
  52.      * 对指定目录进行递归。 
  53.      *  
  54.      * 多级目录下,都要用到相同的集合和过滤器,那么不要在递归方法中定义,而是不断地进行传递。 
  55.      *  
  56.      * @param dir 需要遍历的目录 
  57.      * @param list 用于存储符合条件的File对象 
  58.      * @param filter 接收指定的过滤器 
  59.      */  
  60.     public static void getFileList(File dir, List list, FileFilter filter){  
  61.           
  62.         //1.通过ListFiles方法,获取dir当前下的所有的文件和文件夹对象。  
  63.         File[] files = dir.listFiles();  
  64.           
  65.         //2.遍历该数组。  
  66.         for(File file : files){  
  67.               
  68.             //3.判断是否是文件夹。如果是,递归。如果不是,那就是文件,就需要对文件进行过滤。  
  69.             if (file.isDirectory()){  
  70.                 getFileList(file, list, filter);  
  71.             }else{  
  72.                 //4.通过过滤器对文件进行过滤。  
  73.                 if(filter.accept(file)){  
  74.                     list.add(file);  
  75.                 }  
  76.             }  
  77.         }  
  78.     }  
  79. }  

其中所用到的文件过滤器上面有,这里就不单独再列出来了。

字节流

File文件对象只能操作文件或者文件夹的属性,例如文件或文件夹的创建、删除、获取文件属性(大小、所在目录等),我们最终建立文件的目的,是往文件里面存数据,File对象是做不了这个的。这时,我们就要用到IO流。

InputStream:是表示字节输入流的所有类的超类。
    |--FileInputStream:从文件系统中的某个文件中获得输入字节。哪些文件可用取决于主机环境。
                                   用于读取诸如图像数据之类的原始字节流。要读取字符流,请考虑使用 FileReader。
    |--FilterInputStream:包含其他一些输入流,它将这些流用作其基本数据源,它可以直接传输数据或提供一些额外的功能。
        |--BufferedInputStream:该类实现缓冲的输入流。
        |--DataInputStreamStream:操作基本数据类型值的流。
    |--ObjectInputStream:对象的序列化。
    |--PipedInputStream:管道输出流是管道的发送端。
    |--SequenceInputStream:序列流。
    |--ByteArrayInputStream:操作内存数组。关闭动作无效。
    |--System.in:键盘录入。
OutputStream:此抽象类是表示输出字节流的所有类的超类。
    |--FileoutputStream:文件输出流是用于将数据写入File或FileDescriptor的输出流。
                                      注意处理IO异常。续写和换行。
    |--FilterOutputStream:此类是过滤输出流的所有类的超类。
        |--BufferedOutputStream:该类实现缓冲的输出流。
        |--PrintStream:字节打印流,保证数值的表现形式不变,实现自动刷新和换行。
        |--DataOutputStream:操作基本数据类型值的流。
    |--ObjectOutputStream:对象的反序列化。
    |--PipedOutputStream:管道输入流应该连接到管道输出流。
    |--ByteArrayOutputStream:操作内存数组。关闭动作无效。
    |--System.out:控制台打印到屏幕上。


FileOutputStream

将数据写入到文件中,使用字节输出流:FileOutputStream。

在演示字节输出流之前,有以下三点需要注意:

1.输出流所关联的目的地,如果不存在,会自动创建。如果存在,则替换并覆盖。(这与File对象,如果存在、创建失败有所区别)

2.底层流资源使用完以后一定要记得释放资源。也即IO一定要写finally。

3.一定要在释放资源前先判断输出流对象是否为空。因为try中创建输出流对象失败,则fos依然是null,但是空指针没法调用close()函数释放资源,这回导致抛出NullPointerException异常。

下面演示一下创建字节输出流对象、调用输出流的写功能的代码。

[java]  view plain  copy
  1. package ustc.lichunchun.bytestream;  
  2.   
  3. import java.io.File;  
  4. import java.io.FileOutputStream;  
  5. import java.io.IOException;  
  6.   
  7. public class FileOutputStreamDemo {  
  8.   
  9.     public static void main(String[] args) throws IOException {  
  10.   
  11.         /* 
  12.          * 将数据写入到文件中。 
  13.          * 使用字节输出流。 
  14.          * FileOutputStream。 
  15.          */  
  16.         File dir = new File("tempfile");  
  17.         if(!dir.exists()){  
  18.             dir.mkdir();  
  19.         }  
  20.           
  21.         //1.创建字节输出流对象。用于操作文件,在对象初始化时,必须明确数据存储的目的地。  
  22.         //输出流所关联的目的地,如果不存在,会自动创建。如果存在,则替换并覆盖。(这与File对象,如果存在、创建失败有所区别)  
  23.         FileOutputStream fos = new FileOutputStream("tempfile\\fos.txt");  
  24.           
  25.         //2.调用输出流的写功能。  
  26.         //String str = "abcde";  
  27.         //byte[] buf = str.getBytes();  
  28.         fos.write("abcde".getBytes());  
  29.           
  30.         //3.释放资源。  
  31.         fos.close();  
  32.     }  
  33. }  
接着,我们演示一下带处理IO异常的规范写法:

[java]  view plain  copy
  1. package ustc.lichunchun.bytestream;  
  2.   
  3. import java.io.File;  
  4. import java.io.FileOutputStream;  
  5. import java.io.IOException;  
  6.   
  7. public class IOExceptionDemo {  
  8.     /* 
  9.      *  IO异常的处理方式:IO一定要写finally! 
  10.      */  
  11.     public static void main(String[] args) {  
  12.   
  13.         File dir = new File("tempfile");  
  14.         if(!dir.exists()){  
  15.             dir.mkdir();  
  16.         }  
  17.           
  18.         FileOutputStream fos = null;//如果try中内容失败,fos还是null,所以finally要先判断。  
  19.   
  20.         try {  
  21.             fos = new FileOutputStream("tempfile\\fos.txt");  
  22.   
  23.             fos.write("abcdefg".getBytes());  
  24.   
  25.         } catch (IOException e) {  
  26.             System.out.println(e.toString() + "---");  
  27.   
  28.         } finally {  
  29.             if (fos != null) {// 一定要在释放资源前先判断!  
  30.                 try {  
  31.                     fos.close();  
  32.                 } catch (IOException e) {  
  33.   
  34.                     throw new RuntimeException("关闭失败" + e);//不要把异常抛给main函数、并让主函数声明、虚拟机处理!  
  35.                 }  
  36.             }  
  37.         }  
  38.     }  
  39. }  
需求:实现续写和换行操作。注意:行分隔符可以通过System.getProperty("line.separator");获取。
[java]  view plain  copy
  1. package ustc.lichunchun.bytestream;  
  2.   
  3. import java.io.File;  
  4. import java.io.FileOutputStream;  
  5. import java.io.IOException;  
  6.   
  7. public class NewlineDemo {  
  8.   
  9.     private static final String LINE_SEPARATOR = System.getProperty("line.separator");  
  10.   
  11.     public static void main(String[] args) {  
  12.   
  13.         /* 
  14.          * 续写和换行。 
  15.          *  
  16.          * Linux换行符是"\n" 
  17.          * Windows换行符是"\r\n" 
  18.          * System.getProperty("line.separator") 
  19.          */  
  20.         File dir = new File("tempfile");  
  21.         if(!dir.exists()){  
  22.             dir.mkdir();  
  23.         }  
  24.           
  25.         FileOutputStream fos = null;  
  26.         try{  
  27.             fos = new FileOutputStream("tempfile\\fos.txt",true);//传入true实现续写。  
  28.             String str = LINE_SEPARATOR + "abc";  
  29.             fos.write(str.getBytes());  
  30.         }catch(IOException e){  
  31.             System.out.println(e.toString()+"--");  
  32.         }finally{  
  33.             if(fos != null){  
  34.                 try{  
  35.                     fos.close();  
  36.                 }catch(IOException e){  
  37.                     throw new RuntimeException(""+e);  
  38.                 }  
  39.             }  
  40.         }  
  41.     }  
  42. }  
FileInputStream

那如何将已有文件的数据读取出来呢?既然是读,使用InputStream,而且是要操作文件,FileInpuStream。

为了确保文件一定在读之前是存在的,可以先将字符串路径封装成File对象。

下面演示创建文件字节读取流对象、逐个读取并打印文本文件中的字节。

[java]  view plain  copy
  1. package ustc.lichunchun.bytestream;  
  2.   
  3. import java.io.File;  
  4. import java.io.FileInputStream;  
  5. import java.io.IOException;  
  6.   
  7. public class FileInputStreamDemo {  
  8.   
  9.     public static void main(String[] args) throws IOException {  
  10.   
  11.         /* 
  12.          * 将已有文件的数据读取出来。 
  13.          * 既然是读,使用InputStream 
  14.          * 而且是要操作文件,FileInpuStream。 
  15.          */  
  16.           
  17.         //为了确保文件一定在读之前是存在的,将字符串路径封装成File对象。  
  18.         File file = new File("tempfile\\fos.txt");  
  19.         if(!file.exists()){  
  20.             throw new RuntimeException("要读取的文件不存在");  
  21.         }  
  22.           
  23.         //创建文件字节读取流对象时,必须明确与之关联的数据源。  
  24.         FileInputStream fis = new FileInputStream(file);  
  25.           
  26.         //调用读取流对象的读取方法。read();  
  27.         /*       
  28.         int by1 = fis.read(); 
  29.         System.out.println("by1 = "+by1);//97 
  30.         int by2 = fis.read(); 
  31.         System.out.println("by2 = "+by2);//98 
  32.         int by3 = fis.read(); 
  33.         System.out.println("by3 = "+by3);//99 
  34.         int by4 = fis.read(); 
  35.         System.out.println("by4 = "+by4);//-1 
  36.         int by5 = fis.read(); 
  37.         System.out.println("by5 = "+by5);//-1 
  38.          */   
  39.         int by = 0;  
  40.         while((by = fis.read()) != -1){  
  41.             System.out.println(by);  
  42.         }  
  43.   
  44.         //关闭资源。  
  45.         fis.close();  
  46.     }  
  47. }  
但是你肯定会觉得说,这样一个一个字节的读取文件中的字节,那得有多慢啊?

所以我建议使用下面的这第二种字节流读取方式:创建一个缓冲区字节数组,大小自定义,

然后调用FileInputStream的read(byte[])方法,这样一来,效率会提升不少。

建议这里介绍的三种方法中,选择此种方法。当然,最好是用BufferedInputStream,这我稍后便会阐述。

[java]  view plain  copy
  1. package ustc.lichunchun.bytestream;  
  2.   
  3. import java.io.File;  
  4. import java.io.FileInputStream;  
  5. import java.io.IOException;  
  6.   
  7. public class FileInputStreamDemo2 {  
  8.   
  9.     private static final int DEFAULT_SIZE = 1024*1024*2;//2MB 缓冲区  
  10.   
  11.     public static void main(String[] args) {  
  12.   
  13.         //演示第二种读取方式。read(byte[]); --> 第二种方式较好!  
  14.         File file = new File("tempfile\\fos.txt");  
  15.         if(!file.exists()){  
  16.             throw new RuntimeException("要读取的文件不存在");  
  17.         }  
  18.           
  19.         FileInputStream fis = null;  
  20.         try{  
  21.             fis = new FileInputStream(file);//流与文件关联  
  22.               
  23.             //创建一个缓冲区字节数组。  
  24.             byte[] buf = new byte[DEFAULT_SIZE];//缓冲区大小一般设置为1024的整数倍。  
  25.               
  26.             //调用read(byte[])方法  
  27.             /* 
  28.             int len = fis.read(buf);//len记录的是往字节数组里存储的字节个数 
  29.             System.out.println(len + "..." + new String(buf,0,len));//2...ab 
  30.             int len1 = fis.read(buf); 
  31.             System.out.println(len1 + "..." + new String(buf,0,len1));//1...c 
  32.             int len2 = fis.read(buf); 
  33.             System.out.println(len2 + "..." + new String(buf));//-1...cb 
  34.             */  
  35.               
  36.             int len = 0;  
  37.             while((len = fis.read(buf)) != -1){  
  38.                 System.out.println(new String(buf,0,len));  
  39.             }  
  40.         }catch(IOException e){  
  41.             //一般将异常信息写入到日志文件中,进行记录。  
  42.         }finally{  
  43.             if(fis != null){  
  44.                 try{  
  45.                     fis.close();  
  46.                 }catch(IOException e){  
  47.                     //一般可以throw new RuntimeException异常。或者将异常信息写入到日志文件中,进行记录。  
  48.                 }  
  49.             }  
  50.         }  
  51.     }  
  52. }  
当然了,其实还有一种比较蠢方式,就是把自定义缓冲区的大小设置为和文件本身大小一样大,示例如下。不过我非常不建议用这种方法。

[java]  view plain  copy
  1. package ustc.lichunchun.bytestream;  
  2.   
  3. import java.io.FileInputStream;  
  4. import java.io.IOException;  
  5.   
  6. public class FileInputStreamDemo3 {  
  7.   
  8.     public static void main(String[] args) throws IOException {  
  9.           
  10.         FileInputStream fis = new FileInputStream("tempfile\\fos.txt");  
  11.           
  12.         System.out.println(fis.available());//可以获取与之关联文件的字节数。可以理解为file.length();  
  13.           
  14.         byte[] buf = new byte[fis.available()];//创建了一个和文件大小一样的缓冲区,刚刚好。不建议用。  
  15.         fis.read(buf);  
  16.         String s = new String(buf);  
  17.         System.out.println(s);  
  18.           
  19.         fis.close();  
  20.     }  
  21. }  
需求:复制一个文件。(文件类型不限,可以是文本数据,也可以是媒体数据)

思路:

读取源数据,将数据写到目的中。用到了流,操作设备上的数据。

读,用到输入流;写,用到输出流。而且操作的还是文件。需要用到字节流中操作文件的流对象。

[java]  view plain  copy
  1. package ustc.lichunchun.copy;  
  2.   
  3. import java.io.FileInputStream;  
  4. import java.io.FileOutputStream;  
  5. import java.io.IOException;  
  6.   
  7. public class CopyTextTest {  
  8.   
  9.     public static void main(String[] args) throws IOException {  
  10.           
  11.         /* 
  12.          * 需求:复制一个文件。 
  13.          * 思路: 
  14.          * 读取源数据,将数据写到目的中。 
  15.          * 用到了流,操作设备上的数据。 
  16.          * 读,用到输入流;写,用到输出流。 
  17.          * 而且操作的还是文件。需要用到字节流中操作文件的流对象。 
  18.          */  
  19.         copyText();  
  20.     }  
  21.   
  22.     public static void copyText() throws IOException {  
  23.           
  24.         //1.创建一个输入流和源数据相关联。  
  25.         FileInputStream fis = new FileInputStream("复制文本文件图解.bmp");  
  26.           
  27.         //2.创建一个输出流,并通过输出流创建一个目的。  
  28.         FileOutputStream fos = new FileOutputStream("tempfile\\io_copy.bmp");  
  29.           
  30.         //读一个,写一个。-->这种方式非常不好,效率太低,千万别用此方法。  
  31.         int by = 0;  
  32.         while((by = fis.read()) != -1){  
  33.             fos.write(by);  
  34.         }  
  35.         fos.close();  
  36.         fis.close();  
  37.     }  
  38. }  
很显然,上面这种逐字节读取文件的方式效率太低,所以我们考虑使用在FileInputStream中所讲的第二种方法,自定义缓冲区


[java]  view plain  copy
  1. package ustc.lichunchun.copy;  
  2.   
  3. import java.io.FileInputStream;  
  4. import java.io.FileOutputStream;  
  5. import java.io.IOException;  
  6.   
  7. public class CopyTextByBufTest {  
  8.   
  9.     public static void main(String[] args) {  
  10.         copyTextByBuf();  
  11.     }  
  12.   
  13.     public static void copyTextByBuf() {  
  14.         FileInputStream fis = null;  
  15.         FileOutputStream fos = null;  
  16.         try{  
  17.             fis = new FileInputStream("tempfile\\fos.txt");  
  18.             fos = new FileOutputStream("tempfile\\copy_fos.txt");  
  19.             //创建缓冲区  
  20.             byte[] buf = new byte[1024];//1KB,这就是缓冲区.  
  21.             //定义记录字符个数的变量  
  22.             int len = 0;  
  23.             //循环读写  
  24.             while((len = fis.read(buf)) != -1){  
  25.                 fos.write(buf, 0, len);  
  26.             }  
  27.         }catch(IOException e){  
  28.             //异常日志。  
  29.         }finally{  
  30.             if(fos != null){  
  31.                 try {  
  32.                     fos.close();  
  33.                 } catch (IOException e) {  
  34.                     //异常日志。  
  35.                 }  
  36.             }  
  37.             if(fis != null){  
  38.                 try {  
  39.                     fis.close();  
  40.                 } catch (IOException e) {  
  41.                     //异常日志。  
  42.                 }  
  43.             }  
  44.         }  
  45.     }  
  46. }  
BufferedInputStream、BufferedOutputStream

前面在FileInputStream小节中,我已经介绍了利用自定义缓冲区实现高效的字节流读取。

java也考虑到了这一点,在底层将缓冲区封装成了对象,实际上就是在一个类中封装了数组,

对流所操作的数据进行缓存。缓冲区的作用就是为了提高操作数据的效率。

这样可以避免频繁的在硬盘上寻道操作。缓冲区创建时,必须有被缓冲的流对象与之相关联。

原理图解:


需求:利用缓冲区完成复制图片的例子。

[java]  view plain  copy
  1. package ustc.lichunchun.bytestream.buffer;  
  2.   
  3. import java.io.BufferedInputStream;  
  4. import java.io.BufferedOutputStream;  
  5. import java.io.FileInputStream;  
  6. import java.io.FileOutputStream;  
  7. import java.io.IOException;  
  8.   
  9. public class CopyPicByBufferDemo {  
  10.   
  11.     public static void main(String[] args) throws IOException {  
  12.           
  13.         /* 
  14.          * java将缓冲区封装成了对象,实际上就是在一个类中封装了一个数组,对流所操作的数据进行缓存。 
  15.          * 缓冲区的作用就是为了提高操作数据的效率。这样可以避免频繁的在硬盘上寻道操作。 
  16.          * 缓冲区创建时,必须有被缓冲的流对象。 
  17.          *  
  18.          * 利用缓冲区完成复制图片的例子。 
  19.          */  
  20.         copyPicByBuffer();  
  21.     }  
  22.   
  23.     public static void copyPicByBuffer() throws IOException {  
  24.           
  25.         //演示缓冲区。  
  26.         //1.创建具体的流对象。  
  27.         FileInputStream fis = new FileInputStream("tempfile\\1.jpg");  
  28.           
  29.         //2.让缓冲区与指定流相关联。  
  30.         //对流中的数据进行缓冲。位于内存中的缓冲区的数据读写速度远远大于硬盘上的读写。  
  31.         BufferedInputStream bufis = new BufferedInputStream(fis);//缓冲区默认读8MB字节  
  32.           
  33.         FileOutputStream fos = new FileOutputStream("tempfile\\copy_1.jpg");  
  34.           
  35.         BufferedOutputStream bufos = new BufferedOutputStream(fos);  
  36.           
  37.         byte[] buf = new byte[1024];  
  38.           
  39.         int len = 0;  
  40.           
  41.         while((len = bufis.read(buf)) != -1){  
  42.             bufos.write(buf, 0, len);//使用缓冲区的写入方法将数据先写入到缓冲区中。  
  43.             bufos.flush();//将缓冲区的数据刷新到底层目的地中。(即使不写,缓冲区满了,java也会自动刷新 )  
  44.         }  
  45.   
  46.         //关闭缓冲区,其实关闭的就是被缓冲的流对象。  
  47.         bufos.close();  
  48.         bufis.close();  
  49.     }  
  50. }  
问题探讨:用字节流操作中文数据。

在计算机中,无论是文本,还是图片、mp3、视频等,所有数据最终都是以字节形式存在。

字节流:能操作以字节为单位的文件的流对象。所以字节流能操作计算机上的一切数据。

[java]  view plain  copy
  1. package ustc.lichunchun.readcn;  
  2.   
  3. import java.io.FileInputStream;  
  4. import java.io.FileOutputStream;  
  5. import java.io.IOException;  
  6.   
  7. public class ReadCNDemo {  
  8.   
  9.     public static void main(String[] args) throws IOException {  
  10.           
  11.         /* 
  12.          * 字节流操作中文数据。 
  13.          *  
  14.          * String --> String.getbyte() --> byte[] --> FileOutputStream.write(byte[]) 
  15.          * 这里我们使用的是String类的getBytes()方法按平台默认字符集,编码字符串为字节数组, 
  16.          * 用以将中文字符串按字节流输出到硬盘文件中,并以字节形式存储。 
  17.          *  
  18.          * byte[] --> new String(byte[]) --> String --> System.out.println() 
  19.          * 为了在控制台能够打印出中文字符,而不是逐个字节打印出int值, 
  20.          * 我们又使用String类的String(byte[] bytes,int offset,int length)构造函数按平台默认字符集,解码字节数组为字符串。 
  21.          *  
  22.          * FileInputStream --> read() --> FileOutputStream --> write() 
  23.          * 如果只是将中文文本文件拷贝一份,或者复制其他类型的媒体文件诸如图片音视频等,我们则不必关心各种类型文件所使用的各种编码方式, 
  24.          * 只需要通过输入字节流逐个读取硬盘上文件的字节,然后在通过输出字节流输出到相应目的地文件中即可,相应的特定软件会自行解码并打开的。 
  25.          *  
  26.          * BufferedXxxStream --> 为了高效。 
  27.          *  
  28.          * 在计算机中,无论是文本,还是图片、mp3、视频等,所有数据最终都是以字节形式存在。 
  29.          * 字节流:能操作以字节为单位的文件的流对象。 
  30.          * 所以字节流能操作计算机上的一切数据。 
  31.          * 而字符流只能操作以字符为单位的文本数据。 
  32.          *  
  33.          * 注意: 
  34.          * windows简体中文系统下,默认中文编码表是GBK,其中编码中文时,是两个字节对应一个中文字符。 
  35.          */  
  36.         //writeCNText();  
  37.         readCNText();  
  38.     }  
  39.       
  40.     public static void writeCNText() throws IOException {  
  41.         FileOutputStream fos = new FileOutputStream("tempfile\\cn.txt");  
  42.         fos.write("你A好".getBytes());//按照默认编码表GBK编码,将String编码为byte[]。(System.getProperty("file.encoding")获取)  
  43.         //getBytes():使用平台的默认字符集将此 String编码为 byte 序列,并将结果存储到一个新的 byte 数组中。  
  44.         fos.close();  
  45.     }  
  46.   
  47.     public static void readCNText() throws IOException {  
  48.         FileInputStream fis = new FileInputStream("tempfile\\cn.txt");  
  49.           
  50.         /* 
  51.         //这里对中文文本文件逐个字节读取,并打印到控制台,肯定是不行的,编码方式决定了不可以这样。 
  52.         int by = fis.read(); 
  53.         System.out.println(by);//196 
  54.         int by1 = fis.read(); 
  55.         System.out.println(by1);//227 
  56.         int by2 = fis.read(); 
  57.         System.out.println(by2);//65 
  58.         int by3 = fis.read(); 
  59.         System.out.println(by3);//186 
  60.         int by4 = fis.read(); 
  61.         System.out.println(by4);//195 
  62.         int by5 = fis.read(); 
  63.         System.out.println(by5);//-1 
  64.          */  
  65.           
  66.         //读取中文,按照字节形式。但是一个中文GBK码表中是两个字节。  
  67.         //而字节流的read方法一次只读一个字节。如何可以在控制台获取到一个中文呢?  
  68.         //别读一个就操作,多读一些存起来,再操作。存到字节数组中,将字节数组转成字符串就哦了。  
  69.         //因为String类有一个构造函数可以通过使用平台的默认字符集解码指定的 byte数组,构造一个新的 String。  
  70.         byte[] buf = new byte[1024];  
  71.         int len = 0;  
  72.         len = fis.read(buf);  
  73.         String s = new String(buf,0,len);//将字节数组转成字符串,而且是按照默认的编码表(GBK)进行解码。  
  74.         System.out.println(s);  
  75.         /* 
  76.         //字节是硬盘文件存储基本单位,所以复制文件问题中我不需关心媒体或者中文字符到底几个字节, 
  77.         //我就逐个字节往另一个相同类型文件里存,至于不同编码表,那是打开特定文件的软件的问题,与字节流操作无关。 
  78.         FileOutputStream fos = new FileOutputStream("tempfile\\copy_cn.txt"); 
  79.         int by = 0; 
  80.         while((by = fis.read())!=-1){ 
  81.             fos.write(by); 
  82.         } 
  83.  
  84.         //字节流引发的小问题: 
  85.         //用字节流读取中文太费劲,如果中英文穿插,中文2字节,英文1字节, 
  86.         //如果每次只读一个中文字符new byte[2],然后我就转成字符串打印, 
  87.         //"你"字符可以打印出来,但是A和"好"字符的前一半连在一起出来,就出错了。 
  88.         //也许你会说,我可以加一个判断中英文编码的条件,但是这样做太麻烦。 
  89.         //像上面我们做的那种打印中文文本的程序,需要读取字节数据,再自己调用String构造函数解码,也太麻烦,并且如何解码也不知道。  
  90.         //所以我们就考虑用新的流:字符流。 
  91.         byte[] buf = new byte[2]; 
  92.         int len = 0; 
  93.         while((len = fis.read(buf)) != -1){ 
  94.             System.out.println(new String(buf,0,len)); 
  95.         } 
  96.          */  
  97.         fis.close();  
  98.     }  
  99. }  

正如上面这段代码最后一段的注释中所说,对于纯文本数据的文件,用字节流操作中英文数据时由于编码问题,不同语言的字符对应的字节数不同,会显得比较繁琐。这时我们就该考虑字符流了。但是在这之前,我先把调研到的编码相关背景知识总结分享一下。

编码表

在计算机中,无论是文本,还是图片、mp3、视频等,所有数据最终都是以字节形式存在。字节流:能操作以字节为单位的文件的流对象。所以字节流能操作计算机上的一切数据。文本数据、媒体数据等,什么数据字节流都可以搞定。而字符流只能操作以字符为单位的文本数据

主要的编码表有:ASCII、ISO8859-1、GB2312、GBK、GB18030、Unicode、UTF-8(标识头信息)等。如果想要获取到当前系统使用的默认编码表,可以使用如下代码获取系统平台默认字符集:System.getProperty("file.encoding")--> GBK。

ASCII:美国标准信息交换码,0打头,用一个字节的后7位表示,读一个字节。
ISO8859-1:欧洲码表,用一个字节的8位表示,1打头代表拉丁文,读一个字节。
GB2312:中文编码表,两个字节都是1打头,读两个字节
GBK:扩展版中文编码表,第一个字节1打头,读两个字节10101011 01010110 打头都是1,读到1打头,就读两个字节,再去查GBK表。
Unicode:国际标准码,重新编排码表,统一规定所有文字都是两个字节,Java语言使用的就是unicode。弊端:能用一个字节表示的,都用两个字节装,浪费空间。
UTF-8:对Unicode优化,能用一个字节表示的就不用两个字节,最多用三个字节来表示一个字符。读完相应若干字节后,再去查表。01111010 单字节字符,0打头,读完单字节就去查表;11001010 10101010 两个字节字符,110打头第一个字节,10打头第二个字节,读完两个字节再去查表;11100101 10101001 10010001三个字节字符,1110打头第一个字节,10打头第二、三个字节,读完三个字节再去查表。

字符流为什么不能复制图片?

字符流读完字节数据后,并没有直接往目的地里面写,而是去查表了。但是读到的媒体性字节数据,都有自身千变万化的编码方式,在码表里没有对应内容。那么它就会在表的未知字符区域随便找一些数据写到目的地中,这时,目的数据和源数据就不一致,就不能图片编辑器解析为图片。所以,字符流不能用来操作媒体文件。

编解码详述

编码:字符串 --> 字节数组。(默认都是按照windows系统本地的编码GBK存储的)

解码:字节数组 --> 字符串。

代码示例:

[java]  view plain  copy
  1. package ustc.lichunchun.encode;  
  2.   
  3. import java.io.IOException;  
  4. import java.io.UnsupportedEncodingException;  
  5.   
  6. public class EncodeDemo {  
  7.   
  8.     public static void main(String[] args) throws IOException {  
  9.   
  10.         /* 
  11.          * 字符串-->字节数组:编码。 
  12.          * 字节数组-->字符串:解码。 
  13.          *  
  14.          * 字符串的编码,默认都是按照windows系统本地的编码GBK存储的。 
  15.          * 获取平台默认字符集:System.getProperty("file.encoding"); 
  16.          * (注:字符类型char,在java底层是以unicode码完成的,作为中间转换码,可以忽略不用考虑。) 
  17.          *  
  18.          * 你好: 
  19.          * GBK: -60 -29 -70 -61       一个中文两个字节,每个字节打头的都是1,所以都是负数。 
  20.          * UTF-8: -28 -67 -96 -27 -91 -67   
  21.          *  
  22.          * 1.如果你编码错了,解不出来。 
  23.          * 2.如果编对了,解错了,有可能还有救。 
  24.          *   再次按错误的码表编码,获取原字节,然后再选择另一种码表解码即可。 
  25.          *  
  26.          * 应用:tomcat服务器提供的Web服务使用的是iso8859-1在服务器端编解码。 
  27.          */  
  28.         test1();//编解码示例  
  29.           
  30.         //test2();//有时有救  
  31.           
  32.         //test3();//有时没救  
  33.           
  34.         //test4();//移动联通问题  
  35.     }  
  36.       
  37.     public static void test1() throws UnsupportedEncodingException {  
  38.         String str = "你好";  
  39.           
  40.         //编码。  
  41.         //byte[] buf = str.getBytes();//使用平台默认字符集gbk编码  
  42.         //byte[] buf = str.getBytes("gbk");  
  43.         byte[] buf = str.getBytes("utf-8");  
  44.           
  45.         //printBytes(buf);  
  46.           
  47.         //解码。   
  48.         //String s1 = new String(buf);//使用平台默认字符集gbk解码  
  49.         //String s1 = new String(buf, "gbk");  
  50.         String s1 = new String(buf, "utf-8");  
  51.           
  52.         System.out.println("s1 = "+s1);  
  53.     }  
  54.   
  55.     public static void test2() throws IOException {  
  56.         String str = "你好";  
  57.           
  58.         byte[] buf = str.getBytes("gbk");//11000100 11100011 10111010 11000011  
  59.           
  60.         String s1 = new String(buf, "iso8859-1");//iso都是单字节,所以没有改变原字节。  
  61.           
  62.         System.out.println("s1 = "+s1);//????  
  63.           
  64.         byte[] buf2 = s1.getBytes("iso8859-1");//获取原字节:11000100 11100011 10111010 11000011  
  65.           
  66.         String s2 = new String(buf2,"gbk");  
  67.           
  68.         System.out.println("s2 = "+s2);//你好  
  69.     }  
  70.       
  71.     public static void test3() throws IOException {  
  72.         String str = "你好";//"嘻嘻"、"谢谢"  
  73.           
  74.         byte[] buf = str.getBytes("gbk");  
  75.           
  76.         String s1 = new String(buf, "utf-8");//没有识别出来。以?替代。  
  77.           
  78.         byte[] buf1 = s1.getBytes("gbk");  
  79.           
  80.         System.out.println("s1 = "+s1);//???、????、ππ  
  81.           
  82.         byte[] buf2 = s1.getBytes("utf-8");//获取原字节。只有"谢谢"可以得到。  
  83.           
  84.         printBytes(buf2);//-17 -65 -67 -17 -65 -67 -17 -65 -67  
  85.         //-17 -65 -67 -17 -65 -67 -17 -65 -67 -17 -65 -67   
  86.         //-48 -69 -48 -69  
  87.           
  88.         String s2 = new String(buf2,"gbk");  
  89.           
  90.         System.out.println("s2 = "+s2);  
  91.           
  92.         /* 
  93.         见API文档关于utf-8修改版的编码规范。 
  94.         你好: 
  95.             gbk编码:11000100 11100011 10111010 11000011 
  96.             utf-8解码:??? 
  97.             utf-8编码:11101111 10111111 10111101 11101111 10111111 10111101 11101111 10111111 10111101 
  98.         嘻嘻: 
  99.             gbk编码:11001110 11111011 11001110 11111011 
  100.             utf-8解码:???? 
  101.             utf-8编码:11101111 10111111 10111101 11101111 10111111 10111101 11101111 10111111 10111101 11101111 10111111 10111101 
  102.         谢谢: 
  103.             gbk编码:11010000 10111011 11010000 10111011 
  104.             utf-8解码:ππ 
  105.             utf-8编码:11010000 10111011 11010000 10111011 
  106.         所以最后再对utf-8编码后的字节,用gbk解码,"你好"、"嘻嘻"会出现乱码。 
  107.          */  
  108.     }  
  109.       
  110.     public static void test4() throws IOException {  
  111.         String str = "联通";  
  112.         byte[] buf = str.getBytes("gbk");  
  113.         for(byte b : buf){  
  114.             System.out.println(Integer.toBinaryString(b&255));  
  115.         }  
  116.         /* 
  117.         11000001 
  118.         10101010 
  119.         11001101 
  120.         10101000 
  121.         问题: 
  122.         在桌面上创建两个txt文本文件,1.txt写上"移动",2.txt写上"联通"。 
  123.         发现再次打开1.txt完好,2.txt出现乱码。这是为什么呢? 
  124.         回答: 
  125.         记事本保存时,会按照windows平台默认GBK编码文本文件并保存。(ANSI也是本地默认字符集的意思) 
  126.         这可以通过2.txt文本文件大小为4个字节得知。所以编码没有问题。 
  127.         但是通过上面的小程序可以发现,"联通"的二进制码形式上符合utf-8的编码风格规范, 
  128.         所以记事本再次打开时,会按照utf-8的格式解码这个文本文件,所以会出现乱码。 
  129.          */  
  130.     }  
  131.   
  132.     public static void printBytes(byte[] buf) {  
  133.         for(byte b : buf){  
  134.             System.out.print(b+" ");  
  135.         }  
  136.         System.out.println();  
  137.     }  
  138. }  

上述代码示例中的test2()函数,我们有如下的处理过程:


这种原理在实际中,也有应用,比如tomcat服务器端,就是有这样的转换:


练习按照字节数截取一个字符串。"abc你好"如果截取到半个中文,舍弃。比如截取4字节,abc,截取5个字节abc你。定义功能实现。(友情提示:GB2312编码的一个中文是两个字节,而且两个字节的最高位都是1,也就是说是一个负数。)

思路
1.提示告诉我中文两个字节都是负数。
2.判断截取的最后一个字节是否是负数。如果不是,直接截取;如果是,就往回判断,前一个是否是负数。并记录住负数的个数。如果连续的负数有奇数个,舍弃最后一个字节;如果连续的负数是偶数个,不舍弃。

[java]  view plain  copy
  1. package ustc.lichunchun.encode;  
  2.   
  3. import java.io.IOException;  
  4.   
  5. public class Test {  
  6.   
  7.     public static void main(String[] args) throws IOException {  
  8.   
  9.         /* 
  10.          * 按照字节数截取一个字符串。"abc你好"如果截取到半个中文,舍弃。比如截取4字节,abc,截取5个字节abc你。 
  11.          * 定义功能实现。 
  12.          * 友情提示:GB2312编码的一个中文是两个字节,而且两个字节的最高位都是1,也就是说是一个负数。 
  13.          *  
  14.          * 思路: 
  15.          * 1.提示告诉我中文两个字节都是负数。 
  16.          * 2.判断截取的最后一个字节是否是负数。 
  17.          *   如果不是,直接截取。 
  18.          *   如果是,就往回判断,前一个是否是负数。并记录住负数的个数。如果连续的负数有奇数个,舍弃最后一个字节。 
  19.          *   如果连续的负数是偶数个,不舍弃。 
  20.          *    
  21.          * 拓展1:GBK扩容后,中文既有负数,又有正数。它只保证第一个字节是负数,第二个字节不保证。此程序依然适用。 
  22.          * 拓展2:如果将字符串编码成utf-8格式,咋办?这时,一个中文3个字节。 
  23.          */  
  24.         //字符串转成字节数组。  
  25.         String str = "a琲bc你好d琲e";  
  26.         byte[] buf = str.getBytes("utf-8");  
  27.           
  28.         for(int i = 1; i <= buf.length; i++){  
  29.             String temp = cutStringByCount2(str,i);  
  30.             System.out.println("截取"+i+"个字节是:"+temp);  
  31.         }  
  32.     }  
  33.   
  34.     public static String cutStringByCount1(String str, int len) throws IOException {  
  35.           
  36.         //1.将字符串转成字符数组。因为要判断截取的字节是否是负数,所以要先有字节。  
  37.         byte[] bytes = str.getBytes("gbk");  
  38.         //2.定义计数器,记录住负数的个数。  
  39.         int count = 0;  
  40.         //3.对字节数组进行遍历。应该从截取长度的最后一个字节开始判断,并往回判断。  
  41.         for(int x = len-1; x >= 0; x--){  
  42.             //4.在遍历过程中,只要满足负数就进行计数。只要不是负数,直接结束遍历。  
  43.             if(bytes[x]<0){  
  44.                 count++;  
  45.             }else{  
  46.                 break;  
  47.             }  
  48.         }  
  49.         //5.对遍历后,计数器的值进行判断,奇数,就舍弃最后字节并将字节数组转成字符串。  
  50.         //偶数,不舍弃,将字节数组转成字符串。  
  51.         if(count%2 == 0)  
  52.             return new String(bytes, 0, len);  
  53.         else  
  54.             return new String(bytes, 0, len-1);  
  55.     }  
  56.       
  57.     public static String cutStringByCount2(String str, int len) throws IOException {  
  58.         byte[] bytes = str.getBytes("utf-8");  
  59.         int count = 0;  
  60.         for(int x = len-1; x >= 0; x--){  
  61.             if(bytes[x] < 0){  
  62.                 count++;  
  63.             }else{  
  64.                 break;  
  65.             }  
  66.         }  
  67.         if(count%3 ==0)  
  68.             return new String(bytes, 0, len, "utf-8");  
  69.         else if(count%3 == 1)  
  70.             return new String(bytes, 0, len-1"utf-8");  
  71.         else  
  72.             return new String(bytes, 0, len-2"utf-8");  
  73.     }  
  74. }  

字符流

字符流只能用来操作字符数据--文本。但是由于字符流封装了编码部分,所以操作字符比较方便。字符流=字节流+编码表

Reader:用于读取字符流的抽象类。子类必须实现的方法只有 read(char[],int,int)和 close()。
    |--BufferedReader:从字符输入流中读取文本,缓冲各个字符,从而实现字符、数组和行的高效读取。
                                    可以指定缓冲区的大小,或者可使用默认的大小。大多数情况下,默认值就足够大了。
        |--LineNumberReader:跟踪行号的缓冲字符输入流。此类定义了方法 setLineNumber(int)和 getLineNumber(),
                                                 它们可分别用于设置和获取当前行号。
    |--InputStreamReader:是字节流通向字符流的桥梁:它使用指定的 charset读取字节并将其解码为字符。
                                             它使用的字符集可以由名称指定或显式给定,或者可以接受平台默认的字符集。
        |--FileReader:用来读取字符文件的便捷类。此类的构造方法假定默认字符编码和默认字节缓冲区大小都是适当的。
                         要自己指定这些值,可以先在 FileInputStream上构造一个InputStreamReader。
    |--CharArrayReader:没有调用系统流,只是操作内存中的数组。
    |--StringReader:没有调用系统流,只是操作内存中的数组。
Writer:写入字符流的抽象类。子类必须实现的方法仅有 write(char[],int,int)、flush()和 close()。
    |--BufferedWriter:将文本写入字符输出流,缓冲各个字符,从而提供单个字符、数组和字符串的高效写入。
    |--OutputStreamWriter:是字符流通向字节流的桥梁:可使用指定的 charset将要写入流中的字符编码成字节。
                                             它使用的字符集可以由名称指定或显式给定,否则将接受平台默认的字符集。
        |--FileWriter:用来写入字符文件的便捷类。此类的构造方法假定默认字符编码和默认字节缓冲区大小都是可接受的。
                              要自己指定这些值,可以先在 FileOutputStream上构造一个OutputStreamWriter。
    |--PrintWriter:字符打印流,保证数值的表现形式不变,可以对字节流、字符流自动刷新并换行。
    |--CharArrayWriter:没有调用系统流,只是操作内存中的数组。
    |--StringWriter:没有调用系统流,只是操作内存中的数组。


注意:只要是输出字符流,都有查表后的缓冲区。所以它的write()方法后面一定要跟随flush()刷新操作!(即使没写但运行通过,那也是因为close()封装了flush方法)

转换流

通过编码表的演示,读者应该很容易发现针对纯文本文件,用字符流的方式处理更为方便。那就很自然的考虑到,我如何才能将字节流转换成字符流呢?下面我就来介绍一下字节和字符,二者之间转换的桥梁——转换流。

转换流
字节-->字符:解码(看不懂的-->看得懂的):InputStreamReader-->字节通向字符的桥梁,将读到的若干字节进行解码。
字符-->字节:编码(看得懂的-->看不懂的):OutputStreamWriter-->字符通向字节的桥梁,将字符编码成若干字节。

记住:只要是字符输出流,都有查表后的缓冲区。所以它的write()方法后面一定要跟随flush()刷新操作!
缓冲的原因
每次调用 write方法都会导致在给定字符(或字符集)上调用编码转换器,在写入底层输出流之前,得到的这些字节将在缓冲区中累积。中文最终变成字节才能出去,中文变成什么字节呢?识别中文的码表不止一个,没有指定,会按默认码表GBK。这意味着,会把中文先临时存储起来,去查相应的码表,再把查码表得到的字节存起来缓冲一下,再刷新写出去。之前学的字节流,是因为不用进行查码表、缓冲转换,字节该什么样就什么样,直接读写即可。现在需要拿着字符数据去查码表,把查到的字节数据存起来,再写出去。所以,有了缓冲区,就得做刷新。

转换流代码示例:

[java]  view plain  copy
  1. package ustc.lichunchun.charstream;  
  2.   
  3. import java.io.FileInputStream;  
  4. import java.io.FileOutputStream;  
  5. import java.io.IOException;  
  6. import java.io.InputStreamReader;  
  7. import java.io.OutputStreamWriter;  
  8.   
  9. public class TransStreamDemo {  
  10.   
  11.     public static void main(String[] args) throws IOException {  
  12.           
  13.         /* 
  14.          * 需求:通过字符流读取中文数据。 
  15.          *  
  16.          * 字符流 = 字节流 + 编码表。 
  17.          *  
  18.          * Java内置的码表是Unicode编码。见encode包有关编码解码的代码示例。 
  19.          *  
  20.          * 转换流: 
  21.          * 字节-->字符:解码(看不懂的-->看得懂的):InputStreamReader 
  22.          * 字符-->字节:编码(看得懂的-->看不懂的):OutputStreamWriter 
  23.          */  
  24.         //字节通向字符的桥梁,将读到的若干字节进行解码。  
  25.         //readCnText();  
  26.           
  27.         //字符通向字节的桥梁,将字符编码成若干字节。  
  28.         writeCNText();  
  29.     }  
  30.   
  31.     public static void readCnText() throws IOException {  
  32.           
  33.         //1.操作字节流的字符流对象,必须先有字节流。  
  34.         FileInputStream fis = new FileInputStream("tempfile\\cn.txt");  
  35.           
  36.         //2.建立字节向字符的桥梁。(解码)  
  37.         InputStreamReader isr = new InputStreamReader(fis);  
  38.           
  39.         int ch = isr.read();//注意这里read()读取的是单个字符,所以要强转(char)int,才可以打印在控制台  
  40.         System.out.println((char)ch);//(char)20320 = '你'  
  41.         int ch1 = isr.read();  
  42.         System.out.println((char)ch1);//(char)65 = 'A'  
  43.         int ch2 = isr.read();  
  44.         System.out.println((char)ch2);//(char)22909 = '好'  
  45.         int ch3 = isr.read();  
  46.         System.out.println(ch3);//(char)-1 = '?' , 虚拟机返回-1就代表到达文件末尾 了。  
  47.         isr.close();  
  48.     }  
  49.   
  50.     public static void writeCNText() throws IOException {  
  51.           
  52.         //1.创建字节流对象。  
  53.         FileOutputStream fos = new FileOutputStream("tempfile\\GBK.txt");  
  54.           
  55.         //2.创建字符通向字节的桥梁。  
  56.         OutputStreamWriter osw = new OutputStreamWriter(fos);  
  57.           
  58.         //3.使用osw的write方法直接写中文字符串。因为需要拿着字符数据去查表,所以写入字节数据前,都会存储到缓冲区中。  
  59.         osw.write("你A好");  
  60.           
  61.         //4.需要刷新该字符流的缓冲区。将查表得到的字节数据写到fos流中,然后通过Windows底层资源写入到GBK.txt中。  
  62.         //osw.flush();  
  63.   
  64.         //5.关闭此流前,会先刷新一下。  
  65.         osw.close();  
  66.           
  67.         /* 
  68.         close()底层实现: 
  69.         void close(){ 
  70.             flush(); 
  71.             close(); 
  72.         } 
  73.         */  
  74.     }  
  75. }  
flush和close区别

flush()刷新完,流可以继续使用;
close()刷新完,流直接关闭,流结束了,无法再用。再用,会报Stream closed异常。

转换流的好处:可以指定编码表。

什么时候使用转换流呢?
1.操作文本数据。
2.需要指定编码表。

下面我们使用不同的编码表来演示转换流:

[java]  view plain  copy
  1. package ustc.lichunchun.charstream;  
  2.   
  3. import java.io.FileInputStream;  
  4. import java.io.FileOutputStream;  
  5. import java.io.IOException;  
  6. import java.io.InputStreamReader;  
  7. import java.io.OutputStreamWriter;  
  8.   
  9. public

你可能感兴趣的:(JavaSE)