data:image/s3,"s3://crabby-images/1eb7c/1eb7cfae592472ba98d25212855a592840d82ceb" alt="Java网络编程核心技术详解(视频微课版)"
5.2 创建非阻塞的HTTP服务器
HTTP服务器的主要任务就是接收HTTP请求,然后发送HTTP响应。图5-4是本节所介绍的非阻塞的HTTP服务器范例的模型。
data:image/s3,"s3://crabby-images/e4630/e4630bce726f1e52accf7d6862ade12b2347a39b" alt=""
图5-4 HTTP服务器的对象模型
在这个对象模型中,HttpServer类是服务器主程序,由它启动服务器。AcceptHandler负责接收客户连接,RequestHandler负责接收客户的HTTP请求,对其解析,然后生成相应的HTTP响应,再把它发送给客户。Request类表示HTTP请求,Response类表示HTTP响应,Content类表示HTTP响应的正文。
5.2.1 服务器主程序:HttpServer类
HttpServer类是服务器的主程序,它的实现方式与本书4.3.2节的例程4-3(EchoServer.java)相似,仅启用了单个主线程,采用非阻塞模式来接收客户连接,以及收发数据。例程5-2是HttpServer类的源程序。
例程5-2 HttpServer.java
data:image/s3,"s3://crabby-images/545fc/545fc14bb547ccfa93908e6b96b85841370abd98" alt=""
data:image/s3,"s3://crabby-images/09f2c/09f2c5bf1b5bb6d4cb16e4d76dc522f420262ecf" alt=""
在HttpServer类的service()方法中,当ServerSocketChannel向Selector注册接收连接就绪事件时,设置了一个AcceptHandler附件。
data:image/s3,"s3://crabby-images/042c4/042c46793c5039d36db2d096947161c6276f3f2d" alt=""
AcceptHandler类的handle()方法负责处理接收连接就绪事件。当某种事件发生时,HttpServer类的service()方法从SelectionKey中获得Handler附件,然后调用它的handle()方法。
data:image/s3,"s3://crabby-images/73418/7341847235deb624efc6b2b84f93c903ed0dcc3d" alt=""
5.2.2 具有自动增长的缓冲区的ChannelIO类
自定义的ChannelIO类对SocketChannel进行了包装,增加了自动增长缓冲区容量的功能。当调用socketChannel.read(ByteBuffer buffer)方法时,如果buffer已满(即position=limit),那么即使通道中还有未接收的数据,read方法也不会读取任何数据,而是直接返回0,表示读到了零字节。
为了能读取通道中的所有数据,必须保证缓冲区的容量足够大。在ChannelIO类中,有一个requestBuffer变量,它用来存放客户的HTTP请求数据,当requestBuffer剩余容量已经不足5%,并且还有HTTP请求数据未接收时,ChannelIO会自动扩充requestBuffer的容量,该功能由resizeRequestBuffer()方法完成。
例程5-3是ChannelIO类的源程序,它的read()方法和write()方法利用SocketChannel来接收和发送数据,并且它还提供了实用方法transferTo(),该方法能把文件中的数据发送到SocketChannel中。
例程5-3 ChannelIO.java
data:image/s3,"s3://crabby-images/56410/56410f63de67436afdaa54cabfb7a1b01bc96aac" alt=""
data:image/s3,"s3://crabby-images/41e2b/41e2be6061b59825485c8da67c85ba450fe8b287" alt=""
5.2.3 负责处理各种事件的Handler接口
Handler接口负责处理各种事件,它的定义如下。
data:image/s3,"s3://crabby-images/b769a/b769a6779cb78116d769a9d3064eb94f57a97ff2" alt=""
Handler接口有AcceptHandler和RequestHandler两个实现类。AcceptHandler负责处理接收连接就绪事件,RequestHandler负责处理读就绪和写就绪事件。更确切地说,RequestHandler负责接收客户的HTTP请求,以及发送HTTP响应。
5.2.4 负责处理接收连接就绪事件的AcceptHandler类
AcceptHandler负责处理接收连接就绪事件。它获得与客户连接的SocketChannel,然后向Selector注册读就绪事件,并且创建了一个RequestHandler,把它作为SelectionKey的附件。当读就绪事件发生时,将由这个RequestHandler来处理该事件。例程5-4是AcceptHandler类的源程序。
例程5-4 AcceptHandler.java
data:image/s3,"s3://crabby-images/f586b/f586b4ca648b56bff451d76cc63ca3425235526e" alt=""
data:image/s3,"s3://crabby-images/7491d/7491d65a44f2813c4a557ae594a453a242a64d72" alt=""
在以上AcceptHandler的handle()方法中,还创建了一个ChannelIO,RequestHandler与它关联。RequestHandler会利用ChannelIO来接收和发送数据。
5.2.5 负责接收HTTP请求和发送HTTP响应的RequestHandler类
RequestHandler先通过ChannelIO来接收HTTP请求,当接收到了HTTP请求的所有数据后,就对HTTP请求数据进行解析,创建相应的Request对象,然后依据客户的请求内容,创建相应的Response对象,最后发送Response对象中包含的HTTP响应数据。为了简化程序,RequestHandler仅仅支持GET和HEAD这两种请求方式。例程5-5是RequestHandler的源程序。
例程5-5 RequestHandler.java
data:image/s3,"s3://crabby-images/8f85d/8f85d5b9e39bb362e414c2ef1c5df204a0b9272a" alt=""
data:image/s3,"s3://crabby-images/5fffd/5fffd928cee4ec4482e2876fb843a487e9cab021" alt=""
data:image/s3,"s3://crabby-images/a9b93/a9b93f817aad4d05fca38b177dc2661cb486d044" alt=""
5.2.6 代表HTTP请求的Request类
RequestHandler通过ChannelIO读取HTTP请求数据时,这些数据被放在requestByteBuffer中。当HTTP请求的所有数据接收完毕,就要对requestByteBuffer中的数据进行解析,然后创建相应的Request对象。Request对象就表示特定的HTTP请求。
本范例仅支持GET和HEAD请求方式,在这两种方式下,HTTP请求没有正文部分,并且以“\r\n\r\n”结尾。Request类有3个成员变量:action、uri和version,它们分别表示HTTP请求中的请求方式、URI和HTTP的版本。例程5-6是Request类的源程序。
例程5-6 Request.java
data:image/s3,"s3://crabby-images/ec2e4/ec2e4a1afdb2281875cf9f40256ec31097a49517" alt=""
data:image/s3,"s3://crabby-images/aebaf/aebaf369f5f09e3ca46433542def105210cbf2ab" alt=""
data:image/s3,"s3://crabby-images/3dbbf/3dbbf983bfcaeb36ea5fc6a4614d196838458e3f" alt=""
5.2.7 代表HTTP响应的Response类
Response类表示HTTP响应。它有3个成员变量:code、headerBuffer和content,它们分别表示HTTP响应中的状态代码、响应头和正文。Response类的prepare()方法负责准备HTTP响应的响应头和正文内容,send()方法负责发送HTTP响应的所有数据。例程5-7是Response类的源程序。
例程5-7 Response.java
data:image/s3,"s3://crabby-images/7f44d/7f44d9ebde99e1d3b9251ec36cdae3dbf336ce2f" alt=""
data:image/s3,"s3://crabby-images/dca78/dca78430cce249c4e9260bc4e89ae102e49c9d44" alt=""
data:image/s3,"s3://crabby-images/7ebe7/7ebe7da6b59aec538fb337f00dafc032478d94b3" alt=""
5.2.8 代表响应正文的Content接口及其实现类
Response类有一个成员变量content,表示响应正文,它被定义为Content类型。
data:image/s3,"s3://crabby-images/f3466/f346621f09d863a2ee8bf9992a2ee5004aa6e3b8" alt=""
Content接口表示响应正文,它的定义如下。
data:image/s3,"s3://crabby-images/c96a3/c96a32b315f0c2cefdf5f6ec946993a349d5a566" alt=""
Content接口继承了Sendable接口,Sendable接口表示服务器端可发送给客户的内容,它的定义如下:
data:image/s3,"s3://crabby-images/b1ebc/b1ebcbd39b323abdc147f315e77e07699695d0a0" alt=""
Content接口有StringContent和FileContent两个实现类。StringContent表示字符串形式的正文,FileContent表示文件形式的正文。例如在RequestHandler类的build()方法中,如果HTTP请求方式不是GET和HEAD,就创建一个包含StringContent的Response对象,否则就创建一个包含FileContent的Response对象。
data:image/s3,"s3://crabby-images/8d7c2/8d7c26d9276e5dc26c1b79b4a3a0e22d294e9d1d" alt=""
下面主要介绍FileContent类的实现。FileContent类有一个成员变量fileChannel,它表示读文件的通道。FileContent类的send()方法把fileChannel中的数据发送到ChannelIO的SocketChannel中,如果文件中的所有数据发送完毕,send()方法就返回false。例程5-8是FileContent类的源程序。
例程5-8 FileContent.java
data:image/s3,"s3://crabby-images/cd920/cd92097e831b258009eaf502b1111025b63e440b" alt=""
data:image/s3,"s3://crabby-images/ac28d/ac28d7c5796986a69c7e9eee7621b51bae255801" alt=""
data:image/s3,"s3://crabby-images/80766/807667f19ce788385f4ef068bcada94ffa98cdf2" alt=""
5.2.9 运行HTTP服务器
在chapter05目录下运行命令“java HttpServer”,就启动了HTTP服务器。在本范例的root目录下存放了各种供浏览器访问的文档,比如login.htm、hello.htm和data.rar文件等。打开IE浏览器,输入URL:http://localhost/login.htm或者http://localhost/data.rar,就能接收到服务器发送过来的相应文档。如果浏览器按照POST方式访问hello.htm,服务器就会返回HTTP405错误,因为本服务器不支持POST方式。