💎一站式轻松地调用各大LLM模型接口,支持GPT4、智谱、星火、月之暗面及文生图 广告
## Java编程那些事儿92——IO使用注意问题 陈跃峰 出自:[http://blog.csdn.net/mailbomb](http://blog.csdn.net/mailbomb) ### 11.3.4 注意问题 上面介绍了IO类的基本使用,熟悉了实体流和装饰流的基本使用,但是在IO类实际使用时,还是会遇到一系列的问题,下面介绍一些可能会经常遇到的问题。 #### 11.3.4.1 类的选择 对于初次接触IO技术的初学者来说,IO类体系博大精深,类的数量比较庞大,在实际使用时经常会无所适从,不知道该使用那些类进行编程,下面介绍一下关于IO类选择的一些技巧。 选择类的第一步是选择合适的实体流。 选择实体流时第一步是按照连接的数据源种类进行选择,例如读写文件应该使用文件流,如FileInputStream/FileOutputStream、FileReader/FileWriter,读写字节数组应该使用字节数组流等,如ByteArrayInputStream/ByteArrayOutputStream。 选择实体流时第二步是选择合适方向的流。例如进行读操作时应该使用输入流,进行写操作时应该使用输出流。 选择实体流时第三步是选择字节流或字符流。除了读写二进制文件,或字节流中没有对应的流时,一般都优先选择字符流。 经过以上步骤以后,就可以选择到合适的实体流了。下面说一下装饰流的选择问题。 在选择IO类时,实体流是必需的,装饰流是可选的。另外在选择流时实体流只能选择一个,而装饰流可以选择多个。 选择装饰流时第一步是选择符合要求功能的流。例如需要缓冲流的话选择BufferedReader/BufferedWriter等,有些时候也可能只是为了使用某个装饰流内部提供的方法。 选择装饰流时第二步是选择合适方向的流,这个和实体流选择中的第二步一致。 当选择了多个装饰流以后,可以使用流之间的多层嵌套实现要求的功能,流的嵌套之间没有顺序。 #### 11.3.4.2  非依次读取流数据 由于IO类设计的特点,在实际读取时,只能依次读取流中的数据,而且在通常情况下,已经读取过的数据无法再进行读取。如果需要重复读取流中某段数据时,一般的做法是将从流中读取的数据使用数组存储起来,然后根据需要读取数组中的内容即可,但是有些时候,还是有一些特殊的情况的,IO类对于这些都进行了支持。 1、间断性的读取流中的数据 对于某些特殊格式的文件,例如字体文件等,在实际读取数据时不需要顺序进行读取,而只需要根据内容的位置进行读取。这样可以使用流中的skip方法实现。例如: int n = fis.skip(100); 该行代码的作用是,以流fis当前位置为基础,当前位置可以是流中的任何位置,向后跳过100个单位(字节流单位为字节,字符流单位是字符),如果再使用read方法继续读取,就是读取跳跃以后新位置的内容,也就相当于跳过了100个单位的内容。 而实际在使用时,实际真正跳过的单位数量作为skip方法的返回值返回。 2、重复读取流中某段数据 当必须重复读取流中同一段数据时,如果对应的流支持mark(标记)的话,则可以重复读取同一段数据。 下面以重复读取控制台输入流System.in为例子,来介绍mark的使用,示例代码如下: ~~~ import java.io.*; /**  * mark使用示例  */ public class MarkUseDemo {          public static void main(String[] args) {                    byte[] b = new byte[1024];                    try{                             //读取数据                             int data = System.in.read();                             //输出第一个字节的数据                             System.out.println("第一个字节:" + data);                             //判断该流是否支持mark                             if(System.in.markSupported()){                                      //记忆当前位置,可以从当前位置                                      //向后最多读取100个字节                                      System.in.mark(100);                                      //读取数据                                      int n = System.in.read(b);                                      //输出读取到的内容                                      System.out.print("第一次读取到的内容:");                                      for(int i = 0;i < n;i++){                                                System.out.print(b[i] + " ");                                      }                                      System.out.println();                                      //回到标记位置                                      System.in.reset();                                      //重复读取标记位置以后的内容                                      n = System.in.read(b);                                      //输出读取到的内容                                      System.out.print("第二次读取到的内容:");                                      for(int i = 0;i < n;i++){                                                System.out.print(b[i] + " ");                                      }                                      System.out.println();                             }                    }catch(Exception e){                             e.printStackTrace();                    }          } } ~~~ 在该示例中,首先调用System.in流中的read方法,读取流中的第一个字节,并把读取到的数据赋值给data,然后将读取到的第一个字节的数据输出出来。 然后调用System.in流中的markSupported判断该流是否支持mark功能,如果支持的话则markSupported方法将返回true。 如果流System.in支持mark,则标记当前位置,并允许从当前位置开始最多读取后续100个字节的数据,其实IO类内部的只读取这些数据,而不真正从流中将这些数据删除。 后续继续读取流中的数据,如果读取的数据超过100个字节,则mark标记失效,并把读取到的有效数据输出到控制台。 如果需要从标记位置重复读取已经读取过的数据,则只需要调用流对象中的reset方法重置流的位置,使流可以回到mark的位置,如果继续读取的话,则从该位置开始可以向后读取。这样就可以从mark的位置开始,再次读取后续的数据了。 例如在控制台输入123456789,则该程序的执行结果是: 第一个字节:49 第一次读取到的内容:50 51 52 53 54 55 56 57 13 10 第二次读取到的内容:50 51 52 53 54 55 56 57 13 10 其中输入的第一个字节是1,读取时该字符的编码是49,而后续的内容就是流的结构,其中流末尾的13和10是在输入时,添加的回车和换行字符。 #### 11.3.4.3 中文问题 由于JDK设计时,对于国际化支持比较好,所以JDK在实际实现时支持很多的字符集,这样在进行特定字符集的处理时就需要特别小心了。 其实在进行中文处理时,只需要注意一个原则就可以了,这个原则就是将中文字符转换为byte数组时使用的字符集,需要和把byte数组转换为中文字符串时的字符集保持一致,这样就不会出现中文问题了。 当然,如果不想手动实现字符串和byte数组的转换,可以使用DataInputStream和DataOutputStream中的readUTF和writeUTF实现读写字符串。 ### 11.4 总结 关于IO类的使用,还需要在实际开发过程中多进行使用,从而更深入的体会IO类设计的初衷,并掌握IO类的使用。 另外,IO类是Java中进行网络编程的基础,所以熟悉IO类的使用也是学习网络编程必须的一个基础。