输入和输出是所有应用中所必需的组成部分,通过IO可以读取输入数据以及存储数据到外部设备上。Java中的输入和输出是通过java.io来支持的。下面是本人在学习中的归纳和体会。


1. File类和文件过滤器


    顾名思义,File类中是有关文件的操作。这里必须明确,文件还包括目录。通过文件或目录路径的字符串作为参数,根据传入的是目录路径还是文件名路径可以分别初始化不同的File对象。在File对象中封装了对文件的操作,例如新建文件,删除文件和重命名等方法。但是它并不能访问文件本身的内容,这需要输入输出流来完成。下面是有关File类的代码。

 
 

  1. public class FileTest  
  2. {  
  3.     public static void main(String[] args) throws IOException  
  4.     {  
  5.         //以当前路径来创建一个File对象  
  6.         File file = new File(".");   
  7.         //直接获取文件名,输出一点  
  8.         System.out.println(file.getName());  
  9.         //获取相对路径的父路径可能出错,下面代码输出null  
  10.         System.out.println(file.getParent());  
  11.         //获取绝对路径  
  12.         System.out.println(file.getAbsoluteFile());  
  13.         //获取上一级路径  
  14.         System.out.println(file.getAbsoluteFile().getParent());  
  15.         //在当前路径下创建一个临时文件  
  16.         File tmpFile = File.createTempFile("aaa"".txt", file);  
  17.         //指定当JVM退出时删除该文件  
  18.         tmpFile.deleteOnExit();  
  19.         //以系统当前时间作为新文件名来创建新文件  
  20.         File newFile = new File(System.currentTimeMillis() + "");  
  21.         System.out.println("newFile对象是否存在:" + newFile.exists());  
  22.         //以指定newFile对象来创建一个文件  
  23.         newFile.createNewFile();  
  24.         //以newFile对象来创建一个目录,因为newFile已经存在,  
  25.         //所以下面方法返回false,即无法创建该目录  
  26.         newFile.mkdir();  
  27.         //使用list方法来列出当前路径下的所有文件和路径  
  28.         String[] fileList = file.list();  
  29.         System.out.println("======当前路径下所有文件和路径如下=====");  
  30.         for (String fileName : fileList)  
  31.         {  
  32.             System.out.println(fileName);  
  33.         }  
  34.       
  35.         //listRoots静态方法列出所有的磁盘根路径。  
  36.         File[] roots = File.listRoots();  
  37.         System.out.println("======系统所有根路径如下=====");  
  38.         for (File root : roots)  
  39.         {  
  40.             System.out.println(root);  
  41.         }  
  42.     }  
  43. }  

    在上面的代码中,使用list()方法可以返回目录下所有文件的路径和名称。使用文件过滤器,可以根据自己的需要选择返回的文件类型。通过一个实现了FilenameFilter接口的类对象作为list()方法的参数,就可以实现文件过滤的作用。过滤的规则在重写FilenameFilter中的accept()方法中定义。代码如下:

 
 

  1. public class FilenameFilterTest  
  2. {  
  3.     public static void main(String[] args)   
  4.     {  
  5.         File file = new File(".");  
  6.         String[] nameList = file.list(new MyFilenameFilter());  
  7.         for (String name : nameList)  
  8.         {  
  9.             System.out.println(name);  
  10.         }  
  11.     }  
  12. }  
  13. //实现自己的FilenameFilter实现类  
  14. class MyFilenameFilter implements FilenameFilter  
  15. {  
  16.     public boolean accept(File dir, String name)  
  17.     {  
  18.         //如果文件名以.java结尾,或者文件对应一个路径,返回true  
  19.         return name.endsWith(".java")  
  20.             || new File(name).isDirectory();  
  21.     }  
  22. }  

2. 输入输出流


    在Java中,从不同角度定义了输入输出流的分类。以内存作为流的方向基点,可将其分为输入流和输出流,流向内存的成为输入流,流出内存的成为输出流。从数据操作的最小单元来分类,可将其分为字节流和字符流。根据抽象程度分类,可将其分为节点流和处理流,处理流是连接到实际物理节点的节点流的封装,这样既可以不必去关注节点的来源(文件或者数组),用统一的方法去操作,又可以使用更加方面的方法来实现操作。


    在学习中,本人对于输入输出流的总结归纳:


    A. 字节流和字符流其实并没有本质的区别,无非是它们二者的操作单元不同。字节流的基类为InputStream和OutputStream,字符流的基类为Reader和Writer。它们所提供的读写方法一样,区别在于参数,需要对应流类型。


    B. 输入流中的read系列方法是从输入流中读取数据的操作。空参数将返回所读取的字节(字符)。read方法中可以传入数组参数,数组的类型必须和流的数据类型相匹配,相当于一个竹筒,一次可读取最多等于数组容量的数据,并将所读数据装进数组中。


    C. 输出流中的write系列方法是向输出流中写入数据的操作。方法中的参数可以是整型或字符型变量,也可以传入数组参数,数组的类型必须和流的数据类型相匹配,相当于一个竹筒,一次可写入最多等于数组容量的数据。特别地,字符型输出流的write方法可以传入字符串。


    D. InputStream和OutputStream,Reader和Writer这四个类作为Java中输入输出流的抽象基类,并不能直接被初始化使用。Java中提供了一些继承了它们的类,用于实现具体的某种流操作。如:

    a.File前缀系列的类(如FileInputStream),用于对文件类进行流操作,使用文件名作为构造函数的参数;示例代码如下:

输出类:

 
 

  1. public class FileOutputStreamTest  
  2. {  
  3.     public static void main(String[] args) throws IOException  
  4.     {  
  5.         FileInputStream fis = null;  
  6.         FileOutputStream fos = null;  
  7.         try 
  8.         {  
  9.             //创建字节输入流  
  10.             fis = new FileInputStream("FileOutputStreamTest.java");  
  11.             //创建字节输入流  
  12.             fos = new FileOutputStream("newFile.txt");  
  13.             byte[] bbuf = new byte[32];  
  14.             int hasRead = 0;  
  15.             //循环从输入流中取出数据  
  16.             while ((hasRead = fis.read(bbuf)) > 0 )  
  17.             {  
  18.                 //每读取一次,即写入文件输出流,读了多少,就写多少。  
  19.                 fos.write(bbuf , 0 , hasRead);  
  20.             }  
  21.         }  
  22.         catch (IOException ioe)  
  23.         {  
  24.             ioe.printStackTrace();  
  25.         }  
  26.         finally 
  27.         {  
  28.             //使用finally块来关闭文件输入流  
  29.             if (fis != null)  
  30.             {  
  31.                 fis.close();  
  32.             }  
  33.             //使用finally块来关闭文件输出流  
  34.             if (fos != null)  
  35.             {  
  36.                 fos.close();  
  37.             }  
  38.         }  
  39.     }  
  40. }  

输入类:

 
 

  1. public class FileInputStreamTest  
  2. {  
  3.     public static void main(String[] args) throws IOException  
  4.     {  
  5.         //创建字节输入流  
  6.         FileInputStream fis = new FileInputStream("FileInputStreamTest.java");  
  7.         //创建一个长度为1024的“竹筒”  
  8.         byte[] bbuf = new byte[1024];  
  9.         //用于保存实际读取的字节数  
  10.         int hasRead = 0;  
  11.         //使用循环来重复“取水”过程  
  12.         while ((hasRead = fis.read(bbuf)) > 0 )  
  13.         {  
  14.             //取出“竹筒”中水滴(字节),将字节数组转换成字符串输入!  
  15.             System.out.print(new String(bbuf , 0 , hasRead ));  
  16.         }  
  17.         fis.close();  
  18.     }  
  19. }  

    b. 数组前缀系列类(如ByteArrayInputSteam),用于对数组进行流操作,使用数组对象作为构造函数的参数;

    c. 转换流InputStreamReader将字节输入流转换为字符输入流;InputStreamReader将字节输出流转换为字符输出流,代码如下:

 
 

  1. public class KeyinTest  
  2. {  
  3.     public static void main(String[] args)   
  4.     {  
  5.         BufferedReader br = null;  
  6.         try 
  7.         {  
  8.             //将Sytem.in对象转换成Reader对象  
  9.             InputStreamReader reader = new InputStreamReader(System.in);  
  10.             //将普通Reader包装成BufferedReader  
  11.             br = new BufferedReader(reader);  
  12.             String buffer = null;  
  13.             //采用循环方式来一行一行的读取  
  14.             while ((buffer = br.readLine()) != null)  
  15.             {  
  16.                 //如果读取的字符串为"exit",程序退出  
  17.                 if (buffer.equals("exit"))  
  18.                 {  
  19.                     System.exit(1);  
  20.                 }  
  21.                 //打印读取的内容  
  22.                 System.out.println("输入内容为:" + buffer);  
  23.             }  
  24.         }  
  25.         catch (IOException ioe)  
  26.         {  
  27.             ioe.printStackTrace();  
  28.         }  
  29.         //关闭输入流  
  30.         finally 
  31.         {  
  32.             try 
  33.             {  
  34.                 br.close();               
  35.             }  
  36.             catch (IOException ioe)  
  37.             {  
  38.                 ioe.printStackTrace();  
  39.             }  
  40.         }  
  41.     }  
  42. }  

    d. 字符流的子类有String前缀(StringReader和StringWriter)系列类可以对字符串进行操作,代码如下:

 
 

  1. public class StringNodeTest  
  2. {  
  3.     public static void main(String[] args)   
  4.     {  
  5.         String src = "从明天起,做一个幸福的人\n" 
  6.             + "喂马,劈柴,周游世界\n" 
  7.             + "从明天起,关心粮食和蔬菜\n" 
  8.             + "我有一所房子,面朝大海,春暖花开\n" 
  9.             + "从明天起,和每一个亲人通信\n" 
  10.             + "告诉他们我的幸福\n";  
  11.         StringReader sr = new StringReader(src);  
  12.         char[] buffer = new char[32];  
  13.         int hasRead = 0;  
  14.         try 
  15.         {  
  16.             //采用循环读取的访问读取字符串  
  17.             while((hasRead = sr.read(buffer)) > 0)  
  18.             {  
  19.                 System.out.print(new String(buffer ,0 , hasRead));  
  20.             }  
  21.         }  
  22.         catch (IOException ioe)  
  23.         {  
  24.             ioe.printStackTrace();  
  25.         }  
  26.         finally 
  27.         {  
  28.             sr.close();  
  29.         }  
  30.         //创建StringWriter时,实际上以一个StringBuffer作为输出节点  
  31.         //下面指定的20就是StringBuffer的初始长度  
  32.         StringWriter sw = new StringWriter(20);   
  33.         //调用StringWriter的方法执行输出  
  34.         sw.write("我远离了大海,\n");  
  35.         sw.write("看不到春暖花开,\n");  
  36.         sw.write("我只有一只小龟,\n");  
  37.         sw.write("一样可以闻到馥郁花香\n");  
  38.         System.out.println("------下面是sw的字符串节点里的内容------:");  
  39.         //使用toString方法返回StringWriter的字符串节点的内容  
  40.         System.out.println(sw.toString());  
  41.     }  
  42. }  

    d.推回输入流PushbackInputStream和PushbackReader带有一个推回缓冲区,使用unread(数组)方法可以将一个字节/字符数组推回到推回缓冲区中。推回输入流也有read方法,使用read方法读取输入流时,首先从输入缓冲区中读取,读完之后才从输入流中读。代码如下:

 
 

  1. public class PushbackTest  
  2. {  
  3.     public static void main(String[] args)   
  4.     {  
  5.         PushbackReader pr = null;  
  6.         try 
  7.         {  
  8.             //创建一个PushbackReader对象,指定推回缓冲区的长度为64  
  9.             pr = new PushbackReader(new FileReader("PushbackTest.java") , 64);  
  10.             char[] buf = new char[32];  
  11.             //用以保存上次读取的字符串内容  
  12.             String lastContent = "";  
  13.             int hasRead = 0;  
  14.             //循环读取文件内容  
  15.             while ((hasRead = pr.read(buf)) > 0)  
  16.             {  
  17.                 //将读取的内容转换成字符串  
  18.                 String content = new String(buf , 0 , hasRead);  
  19.                 int targetIndex = 0;  
  20.                 //将上次读取的字符串和本次读取的字符串拼起来,查看是否包含目标字符串  
  21.                 //如果包含目标字符串  
  22.                 if ((targetIndex = (lastContent + content).indexOf("new PushbackReader")) > 0)  
  23.                 {  
  24.                     //将本次内容和上次内容一起推回缓冲区  
  25.                     pr.unread((lastContent + content).toCharArray());  
  26.                     //再次读取指定长度的内容(就是目标字符串之前的内容)  
  27.                     pr.read(buf , 0 , targetIndex);  
  28.                     //打印读取的内容  
  29.                     System.out.print(new String(buf , 0 ,targetIndex));  
  30.                     System.exit(0);  
  31.                 }  
  32.                 else 
  33.                 {  
  34.                     //打印上次读取的内容  
  35.                     System.out.print(lastContent);  
  36.                     //将本次内容设为上次读取的内容  
  37.                     lastContent = content;  
  38.                 }  
  39.             }  
  40.         }  
  41.         catch (IOException ioe)  
  42.         {  
  43.             ioe.printStackTrace();  
  44.         }  
  45.         finally 
  46.         {  
  47.             try 
  48.             {  
  49.                 if (pr != null)  
  50.                     pr.close();  
  51.             }  
  52.             catch (IOException ioe)  
  53.             {  
  54.                 ioe.printStackTrace();  
  55.             }  
  56.         }  
  57.     }  
  58. }  

    E. 在Java虚拟机中可以通过exec方法执行其他的应用程序,并返回Proess对象。利用该对象的getErrorStream、getInputStream和getOutputStream方法可以分别获得子进程的错误流,输出流和输入流(方向是以程序角度为基点)。

    F. RandomAccessFile类提供对文件的随机访问,程序可以直接跳转到文件的任何地方来读取数据。它内部提供了与字节流同法相同的读写方法,另外又加入了自由定位文件记录指针的方法seek。在读写操作时以字节为单位。在构造函数中还需加入一个mode参数由于制定读写权限。


    G. 需要注意Scanner对象的使用。Scanner对象用于捕获输入,并将输入内容转换为字符串。

 
 

  1. //使用System.in创建Scanner对象,用于获取标准输入  
  2.         Scanner sc = new Scanner(System.in);  
  3.         PrintStream ps = new PrintStream(  
  4.             new FileOutputStream("out.txt"));  
  5.         //增加下面一行将只把回车作为分隔符  
  6.         sc.useDelimiter("\n");  
  7.         //判断是否还有下一个输入项  
  8.         while(sc.hasNext())  
  9.         {  
  10.             //输出输入项  
  11.             ps.println("键盘输入的内容是:" + sc.next());  
  12.         }  
  13.         ps.close(); 

    对于初学者来说一个容易忽略的地方是,需要区别节点流和处理流构造方法中参数的意义。节点流中,构造方法参数是要读取或者输出地物理节点,是“目的地”或者“始发地”,而处理流的构造函数参数是所要包装的处理流对象,包装之后的操作实际上是间接操作节点流,并未对被包装的节点流本身属性做修改。采用处理流包装后,可以不用管节点流的数据类型,而根据处理流的性质传递在节点流中不能传递的内容,例如Obj;同时还可以增加一些便于操作的方法,比如缓冲区。