這篇文章給大家介紹如何通過(guò)入門demo簡(jiǎn)單了解netty的使用,內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對(duì)大家能有所幫助。
創(chuàng)新互聯(lián)公司堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:成都網(wǎng)站制作、做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時(shí)代的硯山網(wǎng)站設(shè)計(jì)、移動(dòng)媒體設(shè)計(jì)的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!前言
最近做一個(gè)項(xiàng)目:
大概需求: 多個(gè)溫度傳感器不斷向java服務(wù)發(fā)送溫度數(shù)據(jù),該傳感器采用socket發(fā)送數(shù)據(jù);該數(shù)據(jù)以$符號(hào)開(kāi)頭和結(jié)尾,最后將處理的數(shù)據(jù)存入數(shù)據(jù)庫(kù);
我想到的處理方式:采用netty來(lái)接收和處理數(shù)據(jù),然后用mybatis將處理后的數(shù)據(jù)存入數(shù)據(jù)庫(kù);
我在這之前從來(lái)沒(méi)使用過(guò)netty,在網(wǎng)上倒是看到不少關(guān)于netty的文章,如今就趁著這個(gè)項(xiàng)目寫一下我所學(xué)到的東西和遇到的問(wèn)題,又是怎么去解決的;
接下來(lái)的幾篇文章都是圍繞著這個(gè)項(xiàng)目來(lái)寫的;本篇主要寫netty的入門demo;
正文
代碼部分
新建一個(gè)maven項(xiàng)目
首先在pom.xml中導(dǎo)入:
服務(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)聽(tīng)端口關(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
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(); }}
啟動(dòng)netty服務(wù);
好了,到這里就能開(kāi)始接收數(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
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次消息
先啟動(dòng)服務(wù)端,再啟動(dòng)客戶端;
測(cè)試結(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)容太多,就不都貼出來(lái)了j,直接寫結(jié)果吧:
客戶端發(fā)送100次數(shù)據(jù),但是服務(wù)端只收到了28次,然后服務(wù)端向客戶端返回28次數(shù)據(jù),客戶端卻只收到一次;
可以發(fā)現(xiàn)服務(wù)端接收的數(shù)據(jù)不是完整接收的,這里出現(xiàn)了拆包,粘包的問(wèn)題
這里就不討論拆包,粘包了,百度一大堆,相信你也能看明白;
解決粘包,拆包的問(wèn)題
解決拆包粘包的方法有很多:
消息定長(zhǎng),固定每個(gè)消息的固定長(zhǎng)度 在消息末尾使用換行符對(duì)消息進(jìn)行分割,或者使用其他特殊字符來(lái)對(duì)消息進(jìn)行分割; 將消息分為消息頭和消息體,消息頭中包含標(biāo)識(shí)消息總長(zhǎng)度; 更復(fù)雜的,或者其他的協(xié)議。
由于我負(fù)責(zé)的這個(gè)項(xiàng)目戶端發(fā)送是由$開(kāi)始和結(jié)束的數(shù)據(jù),返回的數(shù)據(jù)我也設(shè)置的$結(jié)束,所以我選擇了第二種方法;
只需要在服務(wù)端的DiscardServerHandler中和客戶端的ChannelInitializer中添加幾行相同的代碼就行了;
服務(wù)端:
public class ChildChannelHandler extends ChannelInitializer
客戶端:
在如下的位置添加如下的代碼:
.handler(new ChannelInitializer
測(cè)試結(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
客戶端:
收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息收到信息
解決我所遇到的問(wèn)題了;
總結(jié)
本來(lái)我只需要寫服務(wù)端的代碼的,但是為了更好的演示,所以我寫了客戶端 本篇文章主要就是使用netty發(fā)送和接收數(shù)據(jù),還有就是拆包和粘包的問(wèn)題,當(dāng)然,netty還可以做其他很多的事情; netty針對(duì)對(duì)拆包粘包的問(wèn)題有很多種解決辦法:例如可以用LineBasedFrameDecoder和StringDecoder組合將信息已換行符來(lái)進(jìn)行拆分;也可以用我上邊的解決方法來(lái)解決以特殊字符結(jié)束的信息; 在解決拆包粘包信息的時(shí)候,注意信息是否符合定義的規(guī)則,不然會(huì)處理不了數(shù)據(jù):例如我上邊的例子,如果服務(wù)端在返回信息是不以$符結(jié)尾的話,客戶端是打印不出來(lái)信息的,因?yàn)榭蛻舳藭?huì)認(rèn)為服務(wù)端還沒(méi)有發(fā)送完信息,會(huì)一直等待,而且打印不出數(shù)據(jù); 這篇文章只是我入門netty的一個(gè)小demo,對(duì)我還是很有幫助的,當(dāng)然也希望對(duì)閱讀者有那么一點(diǎn)點(diǎn)幫助;
關(guān)于如何通過(guò)入門demo簡(jiǎn)單了解netty的使用就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到。