(92)IO使用注意问题
最后更新于:2022-04-01 11:00:50
## 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类的使用也是学习网络编程必须的一个基础。