欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Netty粘包問(wèn)題的常見(jiàn)解決方案

 更新時(shí)間:2024年06月04日 08:37:57   作者:Java中文社群  
粘包和拆包問(wèn)題也叫做粘包和半包問(wèn)題,它是指在數(shù)據(jù)傳輸時(shí),接收方未能正常讀取到一條完整數(shù)據(jù)的情況(只讀取了部分?jǐn)?shù)據(jù),或多讀取到了另一條數(shù)據(jù)的情況)就叫做粘包或拆包問(wèn)題,本文介紹了Netty如何解決粘包問(wèn)題,需要的朋友可以參考下

引言

粘包和拆包問(wèn)題也叫做粘包和半包問(wèn)題,它是指在數(shù)據(jù)傳輸時(shí),接收方未能正常讀取到一條完整數(shù)據(jù)的情況(只讀取了部分?jǐn)?shù)據(jù),或多讀取到了另一條數(shù)據(jù)的情況)就叫做粘包或拆包問(wèn)題。

從嚴(yán)格意義上來(lái)說(shuō),粘包問(wèn)題和拆包問(wèn)題屬于兩個(gè)不同的問(wèn)題,接下來(lái)我們分別來(lái)看。

1.粘包問(wèn)題

粘包問(wèn)題是指在網(wǎng)絡(luò)通信中,發(fā)送方連續(xù)發(fā)送的多個(gè)小數(shù)據(jù)包被接收方一次性接收的現(xiàn)象。這可能是因?yàn)榈讓觽鬏攲訁f(xié)議(如 TCP)會(huì)將多個(gè)小數(shù)據(jù)包合并成一個(gè)大的數(shù)據(jù)塊進(jìn)行傳輸,導(dǎo)致接收方在接收數(shù)據(jù)時(shí)一次性接收了多個(gè)數(shù)據(jù)包,造成粘連。

例如以下案例,正常情況下客戶端發(fā)送了兩條消息,分別為“ABC”和“DEF”,那么接收端也應(yīng)該收到兩條消息“ABC”和“DEF”才對(duì),但是接收端卻收到了“ABCD”這樣的消息,這種情況就叫做粘包,如下圖所示:

2.拆包/半包問(wèn)題

拆包問(wèn)題是指發(fā)送方發(fā)送的一個(gè)大數(shù)據(jù)包被接收方拆分成多個(gè)小數(shù)據(jù)包進(jìn)行接收的現(xiàn)象。這可能是因?yàn)榈讓觽鬏攲訁f(xié)議(如 TCP)將一個(gè)大數(shù)據(jù)包拆分成多個(gè)小的數(shù)據(jù)塊進(jìn)行傳輸,導(dǎo)致接收方在接收數(shù)據(jù)時(shí)分別接收了多個(gè)小數(shù)據(jù)包,造成拆開(kāi)。

例如以下案例,客戶端發(fā)送了一條消息“ABC”,而接收端卻收到了“AB”和“C”兩條信息,這種情況就叫做半包,如下圖所示:

PS:大部分情況下我們都把粘包問(wèn)題和拆包問(wèn)題看成同一個(gè)問(wèn)題,所以下文就用粘包問(wèn)題來(lái)替代粘包和拆包問(wèn)題。

3.為什么會(huì)有粘包問(wèn)題?

粘包問(wèn)題通常發(fā)生在 TCP/IP 協(xié)議中,因?yàn)?TCP 是面向連接的傳輸協(xié)議,它是以“流”的形式傳輸數(shù)據(jù)的,而“流”數(shù)據(jù)是沒(méi)有明確的開(kāi)始和結(jié)尾邊界的,所以就會(huì)出現(xiàn)粘包問(wèn)題。

4.常見(jiàn)解決方案

粘包問(wèn)題的常見(jiàn)解決方案有以下 3 種:

  • 固定大小方法:發(fā)送方和接收方固定發(fā)送數(shù)據(jù)大小,當(dāng)字符長(zhǎng)度不夠時(shí)用空字符彌補(bǔ),有了固定大小之后就知道每條消息的具體邊界了,這樣就沒(méi)有粘包的問(wèn)題了。
  • 自定義數(shù)據(jù)協(xié)議(定義數(shù)據(jù)長(zhǎng)度):在 TCP 協(xié)議的基礎(chǔ)上封裝一層自定義數(shù)據(jù)協(xié)議,在自定義數(shù)據(jù)協(xié)議中,包含數(shù)據(jù)頭(存儲(chǔ)數(shù)據(jù)的大?。┖?數(shù)據(jù)的具體內(nèi)容,這樣服務(wù)端得到數(shù)據(jù)之后,通過(guò)解析數(shù)據(jù)頭就可以知道數(shù)據(jù)的具體長(zhǎng)度了,也就沒(méi)有粘包的問(wèn)題了。
  • 特殊分割符:以特殊的字符結(jié)尾,比如以“\n”結(jié)尾,這樣我們就知道數(shù)據(jù)的具體邊界了,從而避免了粘包問(wèn)題。

以上三種方案中,第一種固定大小的方法可能會(huì)造成網(wǎng)絡(luò)流量的浪費(fèi),以及傳輸性能慢的問(wèn)題;第二種解決方案實(shí)現(xiàn)難度大,且不利于維護(hù),所以比較推薦的是第三種方案,使用特殊分隔符來(lái)區(qū)分消息的邊界,從而避免粘包問(wèn)題。

5.Netty解決方案

Netty 解決方案也延續(xù)了上面的常見(jiàn)解決方案,它的解決方案有以下幾個(gè):

  • 使用定長(zhǎng)解碼器(FixedLengthFrameDecoder):每個(gè)數(shù)據(jù)包都擁有固定的長(zhǎng)度,接收端根據(jù)固定長(zhǎng)度對(duì)數(shù)據(jù)進(jìn)行切分,從而解決了粘包問(wèn)題。
  • 使用行分隔符解碼器(LineBasedFrameDecoder):以行為單位進(jìn)行數(shù)據(jù)包的解碼,從而解決粘包問(wèn)題。
  • 使用分隔符解碼器(DelimiterBasedFrameDecoder):使用特定的分隔符來(lái)標(biāo)識(shí)消息邊界,這樣接收端可以根據(jù)分隔符正確切分消息。
  • 使用長(zhǎng)度字段解碼器(LengthFieldBasedFrameDecoder):在消息頭部加入表示消息長(zhǎng)度的字段,接收端根據(jù)長(zhǎng)度字段來(lái)確定消息的邊界,而從解決粘包問(wèn)題。

PS:在 Netty 中,解碼器(Decoder)起著非常重要的作用。解碼器主要負(fù)責(zé)將從網(wǎng)絡(luò)中接收到的原始字節(jié)流數(shù)據(jù)轉(zhuǎn)換為應(yīng)用程序能夠理解的 Java 對(duì)象或消息格式。使用解碼器可以解決粘包和拆包問(wèn)題、協(xié)議轉(zhuǎn)換問(wèn)題、消息編碼(如文本轉(zhuǎn)換為字節(jié)流)等問(wèn)題。

這些解碼器的使用如下。

5.1 定長(zhǎng)解碼器

定長(zhǎng)解碼器(FixedLengthFrameDecoder)使用示例如下:

ChannelPipeline pipeline = ch.pipeline();
// 假設(shè)每條消息長(zhǎng)度為 5
pipeline.addLast(new FixedLengthFrameDecoder(5)); 
pipeline.addLast(new StringDecoder());
pipeline.addLast(new YourBusinessLogicHandler());

5.2 行分隔符解碼器

行分隔符解碼器(LineBasedFrameDecoder)使用示例如下:

ChannelPipeline pipeline = ch.pipeline();
// 設(shè)置行分隔符解碼器最大(幀)長(zhǎng)度為 8192 字節(jié)
pipeline.addLast(new LineBasedFrameDecoder(8192)); 
pipeline.addLast(new StringDecoder()); // 添加字符串解碼器
pipeline.addLast(new SimpleChannelInboundHandler<String>() {
    @Override
    protected void channelRead0(ChannelHandlerContext ctx, String msg) throws Exception {
        System.out.println("接收到消息:" + msg);
    }
});

5.3 分隔符解碼器

分隔符解碼器(DelimiterBasedFrameDecoder)使用示例如下:

ChannelPipeline pipeline = ch.pipeline();
// 使用 \r\n 來(lái)進(jìn)行分隔
ByteBuf delimiter = Unpooled.copiedBuffer("\r\n".getBytes());
pipeline.addLast(new DelimiterBasedFrameDecoder(1024, delimiter));
pipeline.addLast(new StringDecoder());
pipeline.addLast(new YourBusinessLogicHandler());

5.4 長(zhǎng)度字段解碼器

長(zhǎng)度字段解碼器(LengthFieldBasedFrameDecoder)使用示例如下:

ChannelPipeline pipeline = ch.pipeline();
// 設(shè)置最大幀長(zhǎng)度為 1024 字節(jié),長(zhǎng)度字段位于第 0 個(gè)字節(jié),長(zhǎng)度字段占用 4 個(gè)字節(jié)
pipeline.addLast(new LengthFieldBasedFrameDecoder(1024, 0, 4, 0, 4));
pipeline.addLast(new LengthFieldPrepender(4));
pipeline.addLast(new StringDecoder());
pipeline.addLast(new StringEncoder());
pipeline.addLast(new LengthFieldServerHandler());

課后思考

除了定長(zhǎng)解碼器、行分隔符解碼器、分隔符解碼器、長(zhǎng)度字段解碼器之外,Netty 還有其他解決粘包問(wèn)題的解決方案嗎?如何自定義解碼器?

到此這篇關(guān)于Netty粘包問(wèn)題的常見(jiàn)解決方案的文章就介紹到這了,更多相關(guān)Netty粘包問(wèn)題內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 如何解決異步線程導(dǎo)致的traceId為空的問(wèn)題

    如何解決異步線程導(dǎo)致的traceId為空的問(wèn)題

    文章討論了在使用異步線程時(shí),traceId為空的問(wèn)題,并提出了使用線程池的解決方案,作者分享了個(gè)人經(jīng)驗(yàn),并鼓勵(lì)大家參考和支持腳本之家
    2025-02-02
  • Mybatis 如何在配置文件中給實(shí)體類起別名

    Mybatis 如何在配置文件中給實(shí)體類起別名

    這篇文章主要介紹了Mybatis 如何在配置文件中給實(shí)體類起別名的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-07-07
  • Java基礎(chǔ)之二叉搜索樹(shù)的基本操作

    Java基礎(chǔ)之二叉搜索樹(shù)的基本操作

    發(fā)現(xiàn)許多小伙伴還不清楚Java二叉搜索樹(shù)的基本操作,今天特地整理了這篇文章,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)Java的小伙伴很有幫助,需要的朋友可以參考下
    2021-05-05
  • Java使用Hutool+自定義注解實(shí)現(xiàn)數(shù)據(jù)脫敏

    Java使用Hutool+自定義注解實(shí)現(xiàn)數(shù)據(jù)脫敏

    我們?cè)谑褂檬謾C(jī)銀行的時(shí)候經(jīng)常能看到APP上會(huì)將銀行卡的卡號(hào)中間部分給隱藏掉使用 ***** 來(lái)代替,在某些網(wǎng)站上查看一些業(yè)務(wù)密碼時(shí)(例如簽到密碼等)也會(huì)使用 ***** 來(lái)隱藏掉真正的密碼,那么這種方式是如何實(shí)現(xiàn)的呢,本文將給大家介紹使用Hutool+自定義注解實(shí)現(xiàn)數(shù)據(jù)脫敏
    2023-09-09
  • IDEA?Reformat?Code?格式化代碼(詳解)

    IDEA?Reformat?Code?格式化代碼(詳解)

    平時(shí)使用Ctrl+Alt+L可以格式化代碼,idea幫你整理空格,換行等,讓代碼看起來(lái)更整潔,今天通過(guò)本文給大家分享IDEA?Reformat?Code?格式化?的過(guò)程,感興趣的朋友一起看看吧
    2023-11-11
  • SpringBoot加載配置6種方式分析

    SpringBoot加載配置6種方式分析

    這篇文章主要介紹了SpringBoot加載配置6種方式分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-10-10
  • mybatis如何批量更新list對(duì)象

    mybatis如何批量更新list對(duì)象

    這篇文章主要介紹了mybatis如何批量更新list對(duì)象問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • javaweb中Http協(xié)議詳解

    javaweb中Http協(xié)議詳解

    HTTP是hypertext transfer protocol(超文本傳輸協(xié)議)的簡(jiǎn)寫(xiě),它是TCP/IP協(xié)議的一個(gè)應(yīng)用層協(xié)議,用于定義WEB瀏覽器與WEB服務(wù)器之間交換數(shù)據(jù)的過(guò)程。這篇文章主要為大家詳細(xì)介紹了javaweb中的Http協(xié)議,感興趣的小伙伴們可以參考一下
    2016-05-05
  • Lucene fnm索引文件格式源碼解析

    Lucene fnm索引文件格式源碼解析

    這篇文章主要為大家介紹了Lucene fnm索引文件格式源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-03-03
  • Java實(shí)現(xiàn)二分查找算法實(shí)例分析

    Java實(shí)現(xiàn)二分查找算法實(shí)例分析

    這篇文章主要介紹了Java實(shí)現(xiàn)二分查找算法,實(shí)例分析了二分查找算法的原理與相關(guān)實(shí)現(xiàn)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-07-07

最新評(píng)論