为了使读者更好地理解非阻塞I/O,本节给出了一个简单的例子用来演示如何将非阻塞I/O应用到网络程序中。读者可以先不必管这个例子的具体细节。因为这个例子的主要目的并不是讲解非阻塞I/O的使用,而是先让读者对非阻塞I/O有一个笼统的感性认识。在看完这个例子后,读者可能会有很多疑问,在本章后面的部分将会逐渐揭开这些迷团。这个例子的主要功能是访问新浪网,并将新浪网的首页在控制台上输出。
package test; import java.net.*; import java.nio.*; import java.nio.channels.*; import java.io.*; public class FirstNonBlockingIO { public static void main(String[] args) throws Exception { SocketAddress remote = new InetSocketAddress("www.sina.com.cn", 80); SocketChannel channel = SocketChannel.open(remote); String request = "GET / HTTP/1.1\r\n" + "Host:www.sina.com.cn\r\n" + "Connection:close\r\n\r\n"; ByteBuffer header = ByteBuffer.wrap(request.getBytes()); channel.write(header); ByteBuffer buffer = ByteBuffer.allocate(1024); WritableByteChannel out = Channels.newChannel(System.out); while (channel.read(buffer) != -1) { buffer.flip(); out.write(buffer); buffer.clear(); } channel.close(); } } |
执行如下命令:
java test.FirstNonBlockingIO > sina.txt |
打开sina.txt后,会看到如下的文件内容:
HTTP/1.0 200 OK Date: Sun, 01 Apr 2007 06:53:50 GMT Server: Apache/2.0.58 (Unix) Last-Modified: Sun, 01 Apr 2007 06:50:47 GMT Connection: close </body> </html> |
由于新浪网的主页内容太多,因此,为了方便查看程序运行结果,使用输出重定向符“>”将本该输出到控制台的内容输出到sina.txt文件中。从例程7-1可以看出,主要有三点和同步I/O存在差异。
1. 连接服务器(第013行)。使用SocketChannel类,而不是Socket类。
2. 向服务端写数据(第018行)。 使用SocketChannel类中的write方法,而不是OutputStream.
3. 从服务端读数据(第021行)。使用SocketChannel类中的read方法,而不是InputStream.
除了上面的三点外,在本例中还使用了缓冲区来处理输入输出数据。因此,通道(Channels)和缓冲区(Buffers)是学习非阻塞I/O之前必须掌握的知识。在下面的文章等将详细讲解这两部分的内容。