【IT相关】javajdk7学习笔记:InputStream与OutputStream&nb

想活用输入/输出API,一定要先了解Java中如何以串流(Stream)抽象化输入/输出概念,以及InputStream、OutputStream继承架构。如此一来,无论标准输入/输出、文档输入/输出、网络输入/输出、数据库输入/输出等都可用一致的操作进行处理。

串流设计的概念

Java将输入/输出抽象化为串流,数据有来源及目的地,衔接两者的是串流对象。比喻来说,数据就好比水,串流好比水管,通过水管的衔接,水由一端流向另一端,如图1所示。

串流衔接来源与目的地

从应用程序角度来看,如果要将数据从来源取出,可以使用输入串流,如果要将数据写入目的地,可以使用输出串流。在Java中,输入串流代表对象为java.io.InputStream实例,输出串流代表对象为java.io.OutputStream实例。无论数据源或目的地为何,只要设法取得InputStream或OutputStream的实例,接下来操作输入/输出的方式都是一致,无须理会来源或目的地的真正形式,如图2所示。

从应用程序看InputStream与OutputStream

来源与目的地都不知道的情况下,如何撰写程序?听来不可思议,但实际上就是会有这类需求。举个例子来说,可以设计一个通用的dump()方法:

Stream IO.java

package cc.openhome;

import java.io.*;

public class IO {

u 数据来源与目的地

public static void dump(InputStream src, OutputStream dest)

v 客户端要处理异常

throws IOException {

w 尝试自动关闭资源

try (InputStream input = src; OutputStream utput = dest) {

x 尝试每次从来源读取1024字节

byte[] data = new byte[1024];

y 读取数据

int length = -1;

while ((length = input.read(data)) != -1) {

z 写出数据

output.write(data, 0, length);

}

}

}

}

dump()方法接受InputStream与OutputStream实例,分别代表读取数据的来源,以及输出数据的目的地u。在进行InputStream与OutputStream的相关操作时若发生错误,会抛出java.io.IOException异常,在这里不特别处理,而是在dump()方法上声明throws,由调用dump()方法的客户端处理v。

在不使用InputStream与OutputStream时,必须使用close()方法关闭串流。由于InputStream与OutputStream操作了java.io.Closeable接口,其父接口为java.lang.AutoCloseable接口,因此可使用JDK7尝试自动关闭资源语法w。

思考一下,如果不能使用JDK7尝试自动关闭资源语法,那使用try、catch、finally该怎么写?可以参考一下8.2.2节的内容。

每次从InputStream读入的数据,都会先置入byte数组中x,InputStream的read()方法,每次会尝试读入byte数组长度的数据,并返回实际读入的字节,只要不是-1,就表示读取到数据y。可以使用OutputStream的write()方法,指定要写出的byte数组、初始索引与数据长度z。

那么这个dump()方法的来源是什么?不知道。目的地呢。也不知道。dump()方法并没有限定来源或目的地真实形式,而是依赖于抽象的InputStream、OutputStream。如果要将某个文档读入并另存为另一个文档,则可以这么使用:

Stream Copy.java

package cc.openhome;

import java.io.*;

public class Copy {

public static void main(String[] args) throws IOException {

IO.dump(

new FileInputStream(args[0]),

new FileOutputStream(args[1])

);

}

}

这个程序可以由命令行自变量指定读取的文档来源与写出的目的地,例如:

> java cc.openhome.Copy c:/workspace/Main.java C:/workspace/Main.txt

稍后就会介绍串流继承架构,FileInputStream是InputStream的子类,用于衔接文档以读入数据,FileOutputStream是OutputStream的子类,用于衔接文档以写出数据。

如果要从HTTP服务器读取某个网页,并另存为文档,也可以使用这里设计的dump()方法。例如:

Stream Download.java

package cc.openhome;

import java.io.*;

import java.net.URL;

public class Download {

public static void main(String[] args) throws IOException {

URL url = new URL(args[0]);

InputStream src = url.openStream();

OutputStream dest = new FileOutputStream(args[1]);

IO.dump(src, dest);

}

}

虽然没有正式介绍到网络程序设计,不过java.net.URL的使用很简单,只要指定网址,URL实例会自动进行HTTP协议。可以使用openStream()方法取得InputStream实例,代表与网站连接的数据串流。可以这样指定网址下载文档:

> java cc.openhome.Download http://openhome.cc c:/workspace/index.txt

无论来源或目的地实体形式为何,只要想办法取得InputStream或OutputStream,接下来都是调用InputStream或OutputStream的相关方法。例如,使用java.net.ServerSocket接受客户端联机的例子:

ServerSocket server = null;

Socket client = null;

try {

server = new ServerSocket(port);

while(true) {

client = server.accept();

InputStream input = client.getInputStream();

OutputStream utput = client.getOutputStream();

// 接下来就是操作 InputStream、OutputStream 实例了

}

}

catch(IOException ex) {

}

如果将来学到Servlet,想将文档输出至浏览器,也会有类似的操作:

response.setContentType(“application/pdf”);

InputStream in = this.getServletContext()

.getResourceAsStream(“/WEB-INF/jdbc.pdf”);

OutputStream ut = response.getOutputStream();

byte[] data = new byte[1024];

int length = -1;

while((length = in.read(data)) != -1) {

out.write(data, 0, length);

}

更多IT资料请访问:Tore_m_628846_6316_5_1.html”>http://www.shangxueba.com/sTore_m_628846_6316_5_1.html

带着我的相机和电脑,远离繁华,走向空旷。

【IT相关】javajdk7学习笔记:InputStream与OutputStream&nb

相关文章:

你感兴趣的文章:

标签云: