W3Cschool
恭喜您成為首批注冊用戶
獲得88經(jīng)驗值獎勵
在上節(jié)的內(nèi)容中我們完成了echo服務(wù)器的編寫,接下來就讓我們一起來學習Netty中如何寫一個echo的客戶端,這樣才能讓連接客戶端,并完成信息的傳送。
跟寫服務(wù)器一樣,我們提供 ChannelInboundHandler 來處理數(shù)據(jù)。下面例子,我們用 SimpleChannelInboundHandler 來處理所有的任務(wù),需要覆蓋三個方法:
Listing 2.4 ChannelHandler for the client
@Sharable //1
public class EchoClientHandler extends
SimpleChannelInboundHandler<ByteBuf> {
@Override
public void channelActive(ChannelHandlerContext ctx) {
ctx.writeAndFlush(Unpooled.copiedBuffer("Netty rocks!", //2
CharsetUtil.UTF_8));
}
@Override
public void channelRead0(ChannelHandlerContext ctx,
ByteBuf in) {
System.out.println("Client received: " + in.toString(CharsetUtil.UTF_8)); //3
}
@Override
public void exceptionCaught(ChannelHandlerContext ctx,
Throwable cause) { //4
cause.printStackTrace();
ctx.close();
}
}
1.@Sharable
標記這個類的實例可以在 channel 里共享
2.當被通知該 channel 是活動的時候就發(fā)送信息
3.記錄接收到的消息
4.記錄日志錯誤并關(guān)閉 channel
建立連接后該 channelActive() 方法被調(diào)用一次。邏輯很簡單:一旦建立了連接,字節(jié)序列被發(fā)送到服務(wù)器。該消息的內(nèi)容并不重要;在這里,我們使用了 Netty 編碼字符串 “Netty rocks!” 通過覆蓋這種方法,我們確保東西被盡快寫入到服務(wù)器。
接下來,我們覆蓋方法 channelRead0()。這種方法會在接收到數(shù)據(jù)時被調(diào)用。注意,由服務(wù)器所發(fā)送的消息可以以塊的形式被接收。即,當服務(wù)器發(fā)送 5 個字節(jié)是不是保證所有的 5 個字節(jié)會立刻收到 - 即使是只有 5 個字節(jié),channelRead0() 方法可被調(diào)用兩次,第一次用一個ByteBuf(Netty的字節(jié)容器)裝載3個字節(jié)和第二次一個 ByteBuf 裝載 2 個字節(jié)。唯一要保證的是,該字節(jié)將按照它們發(fā)送的順序分別被接收。 (注意,這是真實的,只有面向流的協(xié)議如TCP)。
第三個方法重寫是 exceptionCaught()。正如在 EchoServerHandler (清單2.2),所述的記錄 Throwable 并且關(guān)閉通道,在這種情況下終止 連接到服務(wù)器。
SimpleChannelInboundHandler vs. ChannelInboundHandler
何時用這兩個要看具體業(yè)務(wù)的需要。在客戶端,當 channelRead0() 完成,我們已經(jīng)拿到的入站的信息。當方法返回時,SimpleChannelInboundHandler 會小心的釋放對 ByteBuf(保存信息) 的引用。而在 EchoServerHandler,我們需要將入站的信息返回給發(fā)送者,由于 write() 是異步的,在 channelRead() 返回時,可能還沒有完成。所以,我們使用 ChannelInboundHandlerAdapter,無需釋放信息。最后在 channelReadComplete() 我們調(diào)用 ctxWriteAndFlush() 來釋放信息。詳見第5、6章
客戶端引導需要 host 、port 兩個參數(shù)連接服務(wù)器。
Listing 2.5 Main class for the client
public class EchoClient {
private final String host;
private final int port;
public EchoClient(String host, int port) {
this.host = host;
this.port = port;
}
public void start() throws Exception {
EventLoopGroup group = new NioEventLoopGroup();
try {
Bootstrap b = new Bootstrap(); //1
b.group(group) //2
.channel(NioSocketChannel.class) //3
.remoteAddress(new InetSocketAddress(host, port)) //4
.handler(new ChannelInitializer<SocketChannel>() { //5
@Override
public void initChannel(SocketChannel ch)
throws Exception {
ch.pipeline().addLast(
new EchoClientHandler());
}
});
ChannelFuture f = b.connect().sync(); //6
f.channel().closeFuture().sync(); //7
} finally {
group.shutdownGracefully().sync(); //8
}
}
public static void main(String[] args) throws Exception {
if (args.length != 2) {
System.err.println(
"Usage: " + EchoClient.class.getSimpleName() +
" <host> <port>");
return;
}
final String host = args[0];
final int port = Integer.parseInt(args[1]);
new EchoClient(host, port).start();
}
}
1.創(chuàng)建 Bootstrap
2.指定 EventLoopGroup 來處理客戶端事件。由于我們使用 NIO 傳輸,所以用到了 NioEventLoopGroup 的實現(xiàn)
3.使用的 channel 類型是一個用于 NIO 傳輸
4.設(shè)置服務(wù)器的 InetSocketAddress
5.當建立一個連接和一個新的通道時,創(chuàng)建添加到 EchoClientHandler 實例 到 channel pipeline
6.連接到遠程;等待連接完成
7.阻塞直到 Channel 關(guān)閉
8.調(diào)用 shutdownGracefully() 來關(guān)閉線程池和釋放所有資源
與以前一樣,在這里使用了 NIO 傳輸。請注意,您可以在 客戶端和服務(wù)器 使用不同的傳輸 ,例如 NIO 在服務(wù)器端和 OIO 客戶端。在第四章中,我們將研究一些具體的因素和情況,這將導致 您可以選擇一種傳輸,而不是另一種。
Copyright©2021 w3cschool編程獅|閩ICP備15016281號-3|閩公網(wǎng)安備35020302033924號
違法和不良信息舉報電話:173-0602-2364|舉報郵箱:jubao@eeedong.com
掃描二維碼
下載編程獅App
編程獅公眾號
聯(lián)系方式:
更多建議: