springboot整合netty過程詳解
這篇文章主要介紹了springboot整合netty過程詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
前言
上一篇講了netty的一個入門的demo;項目上我也把數(shù)據(jù)處理做好了,就要開始存數(shù)據(jù)庫了;我用的mybatis框架,如果單獨使用還是覺得比較麻煩,所以就用了springboot+mybatis+netty;本篇主要講netty與springboot的整合,以及我在這個過程中遇到的問題,又是怎么去解決的;
正文
我在做springboot與netty整合的時候在谷歌,百度找了無數(shù)文章,都沒有一篇是自己想要的,也達不到自己所想的目的;
代碼
1. 新建一個springboot項目,在pom文件中添加netty依賴:
<dependency> <groupId>io.netty</groupId> <artifactId>netty-all</artifactId> <version>5.0.0.Alpha1</version> </dependency>
2.新建netty服務(wù)
其實可以復(fù)制上一篇文章的netty的三個服務(wù)類,做一些稍微的修改就行了;這里為了方便演示,且修都是改好了的,就直接貼出來了;
DiscardServer類:
@Component public class DiscardServer { @Resource private ChildChannelHandler childChannelHandler; public void run(int port) throws Exception { EventLoopGroup bossGroup = new NioEventLoopGroup(); EventLoopGroup workerGroup = new NioEventLoopGroup(); System.out.println("準(zhǔn)備運行端口:" + port); try { ServerBootstrap bootstrap = new ServerBootstrap(); bootstrap.group(bossGroup, workerGroup) .channel(NioServerSocketChannel.class) .option(ChannelOption.SO_BACKLOG, 128) .childHandler(childChannelHandler); //綁定端口,同步等待成功 ChannelFuture f = bootstrap.bind(port).sync(); //等待服務(wù)監(jiān)聽端口關(guān)閉 f.channel().closeFuture().sync(); } finally { //退出,釋放線程資源 workerGroup.shutdownGracefully(); bossGroup.shutdownGracefully(); } } }
ChildChannelHandler類
@Component public class ChildChannelHandler extends ChannelInitializer<SocketChannel> { @Resource private DiscardServerHandler discardServerHandler; public void initChannel(SocketChannel socketChannel) throws Exception { socketChannel.pipeline().addLast(discardServerHandler); } }
3.DiscardServerHandler類
特別注意DiscardServerHandler類上需要加@Sharable注解,如果不加的話會報錯;
@Component @Sharable public class DiscardServerHandler extends ChannelHandlerAdapter { @Resource private BaseService baseService; @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)); //這里調(diào)用service服務(wù) baseService.test(); } finally { ReferenceCountUtil.release(msg); } } @Override public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception { // 出現(xiàn)異常就關(guān)閉 cause.printStackTrace(); ctx.close(); } }
3.netty調(diào)用所需的服務(wù)類
1.BaseService接口
public interface BaseService { /** * 測試接口 */ void test(); }
2.接口實現(xiàn)類BaseServiceImpl:
@Service public class BaseServiceImpl implements BaseService { @Override public void test() { System.out.println("調(diào)用service服務(wù)"); } }
4 springboot啟動類
由于main方法是靜態(tài)方法,netty服務(wù)啟動類不是靜態(tài)類,在main方法里面需要用new的方式啟動;
也可以將netty服務(wù)啟動類改為靜態(tài)類,然后調(diào)用其他非靜態(tài)的類時就得用new方法來構(gòu)造其他類了;
我也百度到了幾篇文章說實現(xiàn)CommandLineRunner接口,所以我用了springboot啟動類實現(xiàn)CommandLineRunner接口的run方法,然后在run方法里啟動netty服務(wù)
@SpringBootApplication public class DemoApplication implements CommandLineRunner { @Resource private DiscardServer discardServer; public static void main(String[] args) { SpringApplication.run(DemoApplication.class, args); } @Override public void run(String... args) throws Exception { discardServer.run(8080); } }
5.測試
寫一個能發(fā)送數(shù)據(jù)的socket就可以了;
發(fā)送的數(shù)據(jù)為:
public static void main(String[] args){ try { Socket socket=new Socket("localhost",8080); OutputStream outputStream = socket.getOutputStream(); PrintWriter printWriter=new PrintWriter(outputStream); printWriter.write("$tmb00035ET3318/08/22 11:5804029.94,027.25,20.00,20.00$"); printWriter.flush(); socket.shutdownOutput(); socket.close(); } catch (IOException e) { e.printStackTrace(); } }
我的測試結(jié)果:
傳輸內(nèi)容是 $tmb00035ET3318/08/22 11:5804029.94,027.25,20.00,20.00$ aaaaa
到這里,netty與springboot的整合就完成了;
我在整合過程中遇到的問題
我使用springboot結(jié)合netty的流程
springboot啟動類中啟動netty啟動類(DiscardServer),netty啟動類(DiscardServer)再調(diào)用初始化channel類(ChildChannelHandler),然后初始化channel類再調(diào)用(DiscardServerHandler)類;然后DiscardServerHandler類再調(diào)用service服務(wù);如下示例圖:
問題:
- springboot啟動類我并沒有實現(xiàn)CommandLineRunner接口,直接在main方法通過new的方式啟動netty服務(wù)
- 我實現(xiàn)了CommandLineRunner接口,但是我在run方法中用的new的方式啟動的netty服務(wù)或者我在run方法使用注入的方式啟動netty,但是在其他某個地方調(diào)用另一個類使用了new的方式;
- DiscardServerHandler類上為標(biāo)記@Sharable類,會報錯誤;
以上總結(jié)起來的問題就是我在springboot整合netty的過程中有其中一處的調(diào)用其他類時使用的方式是new構(gòu)造的,這樣雖然springboot能啟動,netty也能啟動,但是netty服務(wù)中使用new構(gòu)造的那個類中無法依賴注入,會報空指針異常;
舉個栗子:在圖中的過程中,我在ChildChannelHandler類中通過new的方式調(diào)用DiscardServerHandler類,其他的過程都是使用注入的方式調(diào)用,就會出現(xiàn)上邊的問題;
在遇到空指針的時候,我把spring托管的bean打印了出來,所有的類都在spring的托管中,但是就是無法注入,我也一直沒有明白怎么回事,最后用了一個極端的方法,就是在調(diào)用服務(wù)時,獲取spring的上下文,然后再根據(jù)名字來獲取bean,你谷歌或百度:非托管類調(diào)用spring托管類,就能找到很多文章了;雖然用這個方式能解決上述的問題,但總是不好的;
最后的解決辦法:所以類之間的調(diào)用都使用spring的依賴注入,別用new的方式來調(diào)用或者靜態(tài)方法的方式調(diào)用
總結(jié)
既然項目中用到了spring,那么類與類之間的調(diào)用就用依賴注入,不然會報空指針的問題(就是非托管對象調(diào)用spring托管對象);這也算是一個常識性的問題了,只是自己現(xiàn)在才遇到這樣的問題,還是要踩坑才能遇漲記性啊;這些問題困擾了我兩三天,還是要有經(jīng)驗的人帶,如果有經(jīng)驗的人帶的話,說不幾分鐘就搞定了;
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Springboot整合Dubbo+Nacos實現(xiàn)RPC調(diào)用的示例代碼
隨著互聯(lián)網(wǎng)技術(shù)的飛速發(fā)展,越來越多的企業(yè)和開發(fā)者開始關(guān)注微服務(wù)架構(gòu),Nacos是阿里巴巴開源的一個動態(tài)服務(wù)發(fā)現(xiàn)、配置管理和服務(wù)管理平臺,本文講解如何將Spring Boot與Dubbo和Nacos整合,實現(xiàn)RPC調(diào)用,需要的朋友可以參考下2024-02-02springboot中使用Hibernate-Validation校驗參數(shù)詳解
這篇文章主要為大家介紹了springboot中使用Hibernate-Validation校驗參數(shù)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-07-07java HashMap,TreeMap與LinkedHashMap的詳解
這篇文章主要介紹了 java HashMap,TreeMap與LinkedHashMap的詳解的相關(guān)資料,這里提供實例代碼,幫助大家學(xué)習(xí)理解 這部分的內(nèi)容,需要的朋友可以參考下2016-11-11Mybatis配置之<environments>配置元素詳解
這篇文章主要介紹了Mybatis配置之<environments>配置元素,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-01-01關(guān)于Java8新特性O(shè)ptional類的詳細(xì)解讀
Optional類是一個容器類,它可以保存類型T的值,代表這個值存在。或者僅僅保存null,表示這個值不存在,原來用 null 表示一個值不存在,現(xiàn)在Optional 可以更好的表達這個概念。并且可以避免空指針異常,需要的朋友可以參考下2023-05-05mybatis模糊查詢之bind標(biāo)簽和concat函數(shù)用法詳解
大家都知道bind 標(biāo)簽可以使用 OGNL 表達式創(chuàng)建一個變量井將其綁定到上下文中,接下來通過本文給大家介紹了mybatis模糊查詢——bind標(biāo)簽和concat函數(shù)用法,需要的朋友可以參考下2022-08-08