如何通過入門demo簡單了解netty的使用-創(chuàng)新互聯(lián)

這篇文章給大家介紹如何通過入門demo簡單了解netty的使用,內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。

創(chuàng)新互聯(lián)公司堅持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:成都網(wǎng)站制作、做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時代的硯山網(wǎng)站設(shè)計、移動媒體設(shè)計的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!

前言

最近做一個項(xiàng)目:

大概需求: 多個溫度傳感器不斷向java服務(wù)發(fā)送溫度數(shù)據(jù),該傳感器采用socket發(fā)送數(shù)據(jù);該數(shù)據(jù)以$符號開頭和結(jié)尾,最后將處理的數(shù)據(jù)存入數(shù)據(jù)庫;

我想到的處理方式:采用netty來接收和處理數(shù)據(jù),然后用mybatis將處理后的數(shù)據(jù)存入數(shù)據(jù)庫;

我在這之前從來沒使用過netty,在網(wǎng)上倒是看到不少關(guān)于netty的文章,如今就趁著這個項(xiàng)目寫一下我所學(xué)到的東西和遇到的問題,又是怎么去解決的;

接下來的幾篇文章都是圍繞著這個項(xiàng)目來寫的;本篇主要寫netty的入門demo;

正文

代碼部分

新建一個maven項(xiàng)目

首先在pom.xml中導(dǎo)入:

<!-- https://mvnrepository.com/artifact/io.netty/netty-all -->    <dependency>      <groupId>io.netty</groupId>      <artifactId>netty-all</artifactId>      <version>5.0.0.Alpha1</version>    </dependency>

服務(wù)端1. DiscardServer類,netty的服務(wù)端

public class DiscardServer {  public void run(int port) throws Exception {    EventLoopGroup bossGroup = new NioEventLoopGroup();    EventLoopGroup workerGroup = new NioEventLoopGroup();    System.out.println("準(zhǔn)備運(yùn)行端口:" + port);    try {      ServerBootstrap b = new ServerBootstrap();      b = b.group(bossGroup, workerGroup)          .channel(NioServerSocketChannel.class)          .option(ChannelOption.SO_BACKLOG, 128)          .childHandler(new ChildChannelHandler());      //綁定端口,同步等待成功      ChannelFuture f = b.bind(port).sync();      //等待服務(wù)監(jiān)聽端口關(guān)閉      f.channel().closeFuture().sync();    } finally {      //退出,釋放線程資源      workerGroup.shutdownGracefully();      bossGroup.shutdownGracefully();    }  }  public static void main(String[] args) throws Exception {    new DiscardServer().run(8080);  }}

2. ChildChannelHandler類:

public class ChildChannelHandler extends ChannelInitializer<SocketChannel> {  protected void initChannel(SocketChannel socketChannel) throws Exception {    socketChannel.pipeline().addLast(new DiscardServerHandler());  }}

3. DiscardServerHandler類

在這里是繼承的ChannelHandlerAdapter類,當(dāng)然還可以繼承其他的類,例如SimpleChannelInboundHandler,ChannelInboundHandlerAdapter都可以

public class DiscardServerHandler extends ChannelHandlerAdapter {  @Override  public void channelRead(ChannelHandlerContext ctx, Object msg) {    try {      ByteBuf in = (ByteBuf) msg;      System.out.println("傳輸內(nèi)容是");      System.out.println(in.toString(CharsetUtil.UTF_8));      ByteBuf resp= Unpooled.copiedBuffer("收到信息$".getBytes());      ctx.writeAndFlush(resp);    } finally {      ReferenceCountUtil.release(msg);    }  }  @Override  public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {    // 出現(xiàn)異常就關(guān)閉    cause.printStackTrace();    ctx.close();  }}

啟動netty服務(wù);

好了,到這里就能開始接收數(shù)據(jù)了;

客服端

1.TimeClient類

public class TimeClient {  public void connect(int port,String host)throws Exception{    //配置客戶端    System.out.println(port+"--"+host);    EventLoopGroup eventLoopGroup=new NioEventLoopGroup();    try {      Bootstrap b=new Bootstrap();      b.group(eventLoopGroup).channel(NioSocketChannel.class)          .option(ChannelOption.TCP_NODELAY,true)          .handler(new ChannelInitializer<SocketChannel>() {            protected void initChannel(SocketChannel socketChannel) throws Exception {              socketChannel.pipeline().addLast(new TimeClientHandler());            }          });      //綁定端口,同步等待成功      ChannelFuture f = b.connect(host,port).sync();      //等待服務(wù)監(jiān)聽端口關(guān)閉      f.channel().closeFuture().sync();    }finally {      //優(yōu)雅退出,釋放線程資源      eventLoopGroup.shutdownGracefully();    }  }  public static void main(String[] args) throws Exception {    new TimeClient().connect(8090,"localhost");  }}

2.TimeClientHandler 類

public class TimeClientHandler extends ChannelHandlerAdapter {  private byte[] req;  public TimeClientHandler(){    req="$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$".getBytes();  }  @Override  public void channelActive(ChannelHandlerContext ctx) throws Exception {    ByteBuf message=null;    for(int i=0;i<100;i++){      message=Unpooled.buffer(req.length);      message.writeBytes(req);      ctx.writeAndFlush(message);    }  }  @Override  public void channelRead(ChannelHandlerContext ctx, Object msg) {    try {      ByteBuf in = (ByteBuf) msg;      System.out.println(in.toString(CharsetUtil.UTF_8));    } finally {      ReferenceCountUtil.release(msg);    }  }  @Override  public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {    // 出現(xiàn)異常就關(guān)閉    cause.printStackTrace();    ctx.close();  }}

在channelActive類中向服務(wù)端發(fā)送100次消息

先啟動服務(wù)端,再啟動客戶端;

測試結(jié)果一:

服務(wù)端:

傳輸內(nèi)容是$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00$$tmb00035ET3318/08/22 11:5704026.7傳輸內(nèi)容是5,027.31,20.00,20.00$$tmb00035ET3318/08/22

客戶端:

8080--localhost收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息

由于內(nèi)容太多,就不都貼出來了j,直接寫結(jié)果吧:

客戶端發(fā)送100次數(shù)據(jù),但是服務(wù)端只收到了28次,然后服務(wù)端向客戶端返回28次數(shù)據(jù),客戶端卻只收到一次;

可以發(fā)現(xiàn)服務(wù)端接收的數(shù)據(jù)不是完整接收的,這里出現(xiàn)了拆包,粘包的問題

這里就不討論拆包,粘包了,百度一大堆,相信你也能看明白;

解決粘包,拆包的問題

解決拆包粘包的方法有很多:

消息定長,固定每個消息的固定長度  在消息末尾使用換行符對消息進(jìn)行分割,或者使用其他特殊字符來對消息進(jìn)行分割;  將消息分為消息頭和消息體,消息頭中包含標(biāo)識消息總長度;  更復(fù)雜的,或者其他的協(xié)議。

由于我負(fù)責(zé)的這個項(xiàng)目戶端發(fā)送是由$開始和結(jié)束的數(shù)據(jù),返回的數(shù)據(jù)我也設(shè)置的$結(jié)束,所以我選擇了第二種方法;

只需要在服務(wù)端的DiscardServerHandler中和客戶端的ChannelInitializer中添加幾行相同的代碼就行了;

服務(wù)端:

public class ChildChannelHandler extends ChannelInitializer<SocketChannel> {  protected void initChannel(SocketChannel socketChannel) throws Exception {    ByteBuf byteBuf= Unpooled.copiedBuffer("$".getBytes());    socketChannel.pipeline().addLast(new DelimiterBasedFrameDecoder(1024,byteBuf));    socketChannel.pipeline().addLast(new DiscardServerHandler());  }}

客戶端:

在如下的位置添加如下的代碼:

.handler(new ChannelInitializer<SocketChannel>() {            protected void initChannel(SocketChannel socketChannel) throws Exception {              ByteBuf byteBuf= Unpooled.copiedBuffer("$".getBytes());              socketChannel.pipeline().addLast(new DelimiterBasedFrameDecoder(1024,byteBuf));              socketChannel.pipeline().addLast(new TimeClientHandler());            }          });

測試結(jié)果

這里我就不發(fā)送100次數(shù)據(jù)了,值發(fā)送10次:

服務(wù)端:

傳輸內(nèi)容是tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00傳輸內(nèi)容是tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00傳輸內(nèi)容是tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00傳輸內(nèi)容是tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00傳輸內(nèi)容是tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00傳輸內(nèi)容是tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00傳輸內(nèi)容是tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00傳輸內(nèi)容是tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00傳輸內(nèi)容是tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00傳輸內(nèi)容是tmb00035ET3318/08/22 11:5704026.75,027.31,20.00,20.00

客戶端:

收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息

解決我所遇到的問題了;

總結(jié)

本來我只需要寫服務(wù)端的代碼的,但是為了更好的演示,所以我寫了客戶端  本篇文章主要就是使用netty發(fā)送和接收數(shù)據(jù),還有就是拆包和粘包的問題,當(dāng)然,netty還可以做其他很多的事情;  netty針對對拆包粘包的問題有很多種解決辦法:例如可以用LineBasedFrameDecoder和StringDecoder組合將信息已換行符來進(jìn)行拆分;也可以用我上邊的解決方法來解決以特殊字符結(jié)束的信息;  在解決拆包粘包信息的時候,注意信息是否符合定義的規(guī)則,不然會處理不了數(shù)據(jù):例如我上邊的例子,如果服務(wù)端在返回信息是不以$符結(jié)尾的話,客戶端是打印不出來信息的,因?yàn)榭蛻舳藭J(rèn)為服務(wù)端還沒有發(fā)送完信息,會一直等待,而且打印不出數(shù)據(jù);  這篇文章只是我入門netty的一個小demo,對我還是很有幫助的,當(dāng)然也希望對閱讀者有那么一點(diǎn)點(diǎn)幫助;

關(guān)于如何通過入門demo簡單了解netty的使用就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

分享題目:如何通過入門demo簡單了解netty的使用-創(chuàng)新互聯(lián)
網(wǎng)頁URL:http://muchs.cn/article16/csjcgg.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站維護(hù)、做網(wǎng)站、移動網(wǎng)站建設(shè)標(biāo)簽優(yōu)化、品牌網(wǎng)站設(shè)計、品牌網(wǎng)站建設(shè)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

微信小程序開發(fā)