连接Netty之构建一个完整的的Netty服务器(一)
写一个 echo 客户端
客户端要做的是:
- 连接服务器
- 发送信息
- 发送的每个信息,等待和接收从服务器返回的同样的信息
- 关闭连接
用 ChannelHandler 实现客户端逻辑
跟写服务器一样,我们提供 ChannelInboundHandler
来处理数据。下面例子,我们用SimpleChannelInboundHandler
来处理所有的任务,需要覆盖三个方法:
channelActive()
- 服务器的连接被建立后调用channelRead0()
- 数据后从服务器接收到调用exceptionCaught()
- 捕获一个异常时调用
2.3 ChannelHandler for the client
|
|
@Sharable
标记这个类的实例可以在 channel 里共享- 当被通知该 channel 是活动的时候就发送信息
- 记录接收到的消息
- 记录日志错误并关闭 channel
建立连接后该 channelActive()
方法被调用一次。逻辑很简单:一旦建立了连接,字节序列被发送到服务器。该消息的内容并不重要;在这里,我们使用了 Netty 编码字符串 “Netty rocks!”通过覆盖这种方法,我们确保东西被尽快写入到服务器。
接下来,我们覆盖方法 channelRead0()
。这种方法会在接收到数据时被调用。注意,由服务器所发送的消息可以以块的形式被接收。即,当服务器发送 5 个字节是不是保证所有的 5 个字节会立刻收到 - 即使是只有 5 个字节,channelRead0()
方法可被调用两次,第一次用一个ByteBuf
( Netty的字节容器) 装载3个字节和第二次一个 ByteBuf
装载 2 个字节。唯一要保证的是,该字节将按照它们发送的顺序分别被接收。 ( 注意,这是真实的,只有面向流的协议如TCP) 。
第三个方法重写是 exceptionCaught()
。正如在 EchoServerHandler ( 例子2.1) ,所述的记录 Throwable
并且关闭通道,在这种情况下终止 连接到服务器。
SimpleChannelInboundHandler vs. ChannelInboundHandler
何时用这两个要看具体业务的需要。在客户端,当channelRead0()
完成,我们已经拿到的入站的信息。当方法返回时,SimpleChannelInboundHandler
会小心的释放对ByteBuf
( 保存信息) 的引用。而在EchoServerHandler,我们需要将入站的信息返回给发送者,由于write()
是异步的,在channelRead()
返回时,可能还没有完成。所以,我们使用ChannelInboundHandlerAdapter
,无需释放信息。最后在channelReadComplete()
我们调用ctxWriteAndFlush()
来释放信息。
引导客户端
客户端引导需要 host 、port 两个参数连接服务器。
2.4 Main class for the client
|
|
1.创建 Bootstrap
2.指定 EventLoopGroup
来处理客户端事件。由于我们使用 NIO 传输,所以用到了NioEventLoopGroup
的实现
3.使用的 channel
类型是一个用于 NIO 传输
4.设置服务器的 InetSocketAddress
5.当建立一个连接和一个新的通道时,创建添加到 EchoClientHandler 实例 到 channelpipeline
6.连接到远程;等待连接完成
7.阻塞直到 Channel
关闭
8.调用 shutdownGracefully()
来关闭线程池和释放所有资源
与以前一样,在这里使用了 NIO 传输。请注意,您可以在 客户端和服务器 使用不同的传输,例如 NIO 在服务器端和 OIO 客户端。在第四章中,我们将研究一些具体的因素和情况,这将导致你可以选择一种传输,而不是另一种。
让我们回顾一下我们在本节所介绍的要点
- 一个
Bootstrap
被创建来初始化客户端 - 一个
NioEventLoopGroup
实例被分配给处理该事件的处理,这包括创建新的连接和处理入站和出站数据 - 一个
InetSocketAddress
为连接到服务器而创建 - 一个 EchoClientHandler 将被安装在
pipeline
当连接完成时 - 之后
Bootstrap.connect()
被调用连接到远程的 - 本例就是 echo(回声)服务器。
编译和运行 Echo 服务器和客户端
编译
本例涉及到多模块 Maven项目的组织,在例子chapter2目录下,执行
|
|
输出如下
2.5 Build Output
|
|
注意事项:
- Maven Reactor 构建顺序:先是 父 POM,然后是子项目
- Netty artifact 没在用户的本地存储库中找到,所以 Maven 就会从互联网上下载
- clean 和 compile 在构建生命周期的运行。事后 mavensurefire-plugin 插件运行,但不会
- 有测试类存在。最后 mavenjar-plugin 执行
这段说明了项目已经成功编译。
运行 Echo 服务器 和 客户端
我们使用 exec-maven-plugin 来运行项目。
在 chapter2/Server 目录,执行
|
|
输出如下:
|
|
在 chapter2/Client 目录,执行
|
|
输出如下:
|
|
在服务器的控制台输出:
|
|
发生了什么事:
- 客户端连接后,它发送消息:“Netty rocks!”
- 服务器输出接收到消息并将其返回给客户端
- 客户输出接收到的消息并退出。
每次运行客户端,你会看到在服务器的控制台输出:
|
|
现在,我们看下错误的情况。在控制台 输入 Ctrl-C 来关闭服务器。而后运行客户端,此时输出如下:
|
|
发生了啥?客户端尝试连接服务器,但服务器是关闭的,所以引发了一个java.net.ConnectException ,这个异常被 EchoClientHandler 的 exceptionCaught() 触发,打印出异常信息,并关闭 channel