Java IO : InputStream、Reader

Java 的输入流主要由InputStream 和Reader 作为基类,而输出流则主要由OutputStream 和 writer 作为基类。

在前面的Java IO:概述中,我们已提到过,流分为字节流和字符流、节点流和处理流。

字节流和字符流

区别在于操作的数据单元不同,字节流操作最小数据单元是8位的字节,字符流操作的最小数据单元是16位的字符。

字节流主要是由InputStream和OutputStream 作为基类,而字符流主要由Reader 和 writer作为基类。

节点流和处理流

从/向一个特定的IO设备(磁盘、网络)读/写数据的流,成为节点流。 如:

对一个已存在的流进行封装或连接,通过封装后流来实现数据读/写功能,称为处理流。

注意:当使用处理流来进行输入/输出时,程序并不会直接连接到实际的数据源。

好处:使用处理流包装节点流,即可以消除不同节点流的实现差异,也可以提供方便的方法完成输入/输出功能。

InputStream 和Reader 都是抽象类,本身不能创建实例,但分别有一个用于读取文件的输入流: FileInputStream 和FileInputReader ,它们都是节点流,即直接和指定文件关联。

InputStream 例子

 

/**
 * java input stream 
 * @author mingx
 *
 */
public class InputStreamExample {

	public static void main(String[] args) {
		
		int count = 0;  // 用于保存实际读取的字节数
		InputStream inputStream = null;

		try {
			inputStream = new FileInputStream(new File("D:\\ZhyTestSpace\\testBase\\src\\test\\java\\testBase\\FileExample.java")); 
			byte[] bbuf = new byte[1024];
			while ((count = inputStream.read(bbuf))  > 0 ) {   //读取文件字节
				System.out.println("取出" + new String(bbuf,0,count));
			}
		} catch (final IOException e) {
			e.printStackTrace();
		} finally {
			try {
				inputStream.close();  // fileInputStream是有缓冲区的,所以用完之后必须关闭,否则可能导致内存占满,数据丢失。 
			} catch (IOException e) {
				e.printStackTrace();
			}
		}

	}

}

FileInputStream 的read() 方法,返回读取到的包含一个字节内容的int变量(0~255),如果read 返回-1,意味着程序已经读到了末尾,此时流内已经没有多余的数据可供读取,此时可以关闭流。 

 

以上程序,创建一个长度为1024字节的字节数组来读取该文件,该文件长度为497字节,所以我们只需要一次read就可以全部读取全部内容。但是如果我们创建较小长度的字节数组,程序运行时可能存在中文乱码问题,例如我们将上面的字节数组定义修改如下:

byte[] bbuf = new byte[10];

输出打印:

       

执行了多次read()方法,并且中文呈乱码格式。

最后程序需要在Finally块中使用inputStream.close来关闭该文件输入流。

Reader实例:

 

public class ReaderExample {

	public static void main(String[] args) throws IOException {
		FileReader fr = null;
		try {
			// 创建字符输入流
			fr = new FileReader("D:\\ZhyTestSpace\\testBase\\src\\test\\java\\testBase\\UserTest.java");
			// 创建一个长度为32的字符数组
			char[] cbuf = new char[32];
			int count = 0;
			while ((count = fr.read(cbuf)) > 0) {
				// 取出字节,将字符数组转换成字符串输入
				System.out.println("取出" + new String(cbuf, 0, count));
			}
		} catch (FileNotFoundException e) {
			e.printStackTrace();
		} finally {
			if (fr != null) {
				fr.close();
			}
		}

	}
}

ReaderExample与前面的InputStream实例无多大区别,这里将字符数组定义32,也就意味着,需要调用多次read()方法才能完全读取输入流的全部数据。

 

最后:InputStream和 Reder都支持如下方法来移动记录指针:

  • void mark (int readAheadLimit) :在记录指针当前位置记录一个标记
  • boolean markSupported() :判断输入流是否支持mark操作
  • void reset():将此流的记录指针重新定位到上一次记录标记(mark)位置。
  • long skip (long n) :记录指针向前移动n个字节/字符

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值