99精品伊人亚洲|最近国产中文炮友|九草在线视频支援|AV网站大全最新|美女黄片免费观看|国产精品资源视频|精彩无码视频一区|91大神在线后入|伊人终合在线播放|久草综合久久中文

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

用epoll來實(shí)現(xiàn)多路復(fù)用

科技綠洲 ? 來源:Linux開發(fā)架構(gòu)之路 ? 作者:Linux開發(fā)架構(gòu)之路 ? 2023-11-09 10:15 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

本人用epoll來實(shí)現(xiàn)多路復(fù)用,epoll觸發(fā)模式有兩種:

  • ET(邊緣模式)
  • LT(水平模式)

LT模式

是標(biāo)準(zhǔn)模式,意味著每次epoll_wait()返回后,事件處理后,如果之后還有數(shù)據(jù),會(huì)不斷觸發(fā),也就是說,一個(gè)套接字上一次完整的數(shù)據(jù),epoll_wait()可能會(huì)返回多次,直到?jīng)]有數(shù)據(jù)為止。

ET模式

也稱高效模式,有數(shù)據(jù)過來后,epoll_wait()會(huì)返回一次,一段時(shí)間內(nèi),該套接字就算有數(shù)據(jù)源源不斷地過來,epoll_wait()也不會(huì)返回了。這里注意,是一段時(shí)間,不代表這個(gè)套接字上有數(shù)據(jù)就只觸發(fā)一次。時(shí)間過長,還是會(huì)返回多次的。比如我寫FTP用了epoll+多線程,但是每次套接字上有信息就開線程處理,同一時(shí)間內(nèi)希望一個(gè)套接字只被一個(gè)線程持有,但是因?yàn)槲募鬏敃r(shí)間過長,就算使用ET模式,套接字還是會(huì)返回多次。這里要特別強(qiáng)調(diào)一個(gè)參數(shù)EPOLLONESHOT,如果要保證套接字同一時(shí)段只被一個(gè)線程處理,必須加上。解決方案:給accept()后的套接字加上參數(shù)EPOLLONESHOT,線程結(jié)束后處理完之后,再重置EPOLLONESHOT屬性,但是,千萬不可以給listen()后的監(jiān)聽套接字設(shè)置此屬性,這會(huì)造成同一時(shí)刻只能處理一個(gè)連接的情況。

深入理解EPOLLONESHOT事件

即使使用ET模式,一個(gè)socket上的某個(gè)事件還是可能被觸發(fā)多次,這是跟數(shù)據(jù)報(bào)的大小有關(guān)系,常見的情景就是一個(gè)線程,而在數(shù)據(jù)的處理過程中該socket上又有新數(shù)據(jù)可讀(EPOLLIN再次被觸發(fā)),此時(shí)另外一個(gè)線程被喚醒處理這些新的數(shù)據(jù),于是出現(xiàn)了兩個(gè)線程同時(shí)操作一個(gè)socket,為了避免這種情況,就可以采用epoll的EPOLLONESPOT事件。同時(shí)要注意,注冊(cè)了EPOLLONESHOT事件的socket一旦被某個(gè)線程處理完畢,該線程就應(yīng)該立即重置這個(gè)socket的EPOLLONESHOT的事件,以確保這個(gè)socket下次可讀時(shí),其EPOLLIN事件被觸發(fā),進(jìn)而讓其他的工作線程有機(jī)會(huì)繼續(xù)處理這個(gè)socket。

網(wǎng)絡(luò)事件EAGIN

在一個(gè)非阻塞的socket上調(diào)用read/write函數(shù), 返回EAGAIN或者EWOULDBLOCK(注: EAGAIN就是EWOULDBLOCK)從字面上看, 意思是:EAGAIN: 再試一次,EWOULDBLOCK: 如果這是一個(gè)阻塞socket, 操作將被block,perror輸出: Resource temporarily unavailable.

小結(jié):

這個(gè)錯(cuò)誤表示資源暫時(shí)不夠,能read時(shí),讀緩沖區(qū)沒有數(shù)據(jù),或者write時(shí),寫緩沖區(qū)滿了。遇到這種情況,如果是阻塞socket,read/write就要阻塞掉。而如果是非阻塞socket,read/write立即返回-1, 同時(shí)errno設(shè)置為EAGAIN。所以,對(duì)于阻塞socket,read/write返回-1代表網(wǎng)絡(luò)出錯(cuò)了。但對(duì)于非阻塞socket,read/write返回-1不一定網(wǎng)絡(luò)真的出錯(cuò)了??赡苁荝esource temporarily unavailable。這時(shí)你應(yīng)該再試,直到Resource available。

EAGAIN: 再試一次,EWOULDBLOCK: 如果這是一個(gè)阻塞socket, 操作將被block,perror輸出: Resource temporarily unavailable。這個(gè)錯(cuò)誤表示資源暫時(shí)不夠,能read時(shí),讀緩沖區(qū)沒有數(shù)據(jù),或者write時(shí),寫緩沖區(qū)滿了。遇到這種情況,如果是阻塞socket,read/write就要阻塞掉。而如果是非阻塞socket,read/write立即返回-1, 同時(shí)errno設(shè)置為EAGAIN。所以,對(duì)于阻塞socket,read/write返回-1代表網(wǎng)絡(luò)出錯(cuò)了。但對(duì)于非阻塞socket,read/write返回-1不一定網(wǎng)絡(luò)真的出錯(cuò)了??赡苁荝esource temporarily unavailable。這時(shí)你應(yīng)該再試,直到Resource available。

綜上,對(duì)于non-blocking的socket,正確的讀寫操作為: 讀:忽略掉errno = EAGAIN的錯(cuò)誤,下次繼續(xù)讀 寫:忽略掉errno = EAGAIN的錯(cuò)誤,下次繼續(xù)寫

對(duì)于select和epoll的LT模式,這種讀寫方式是沒有問題的。但對(duì)于epoll的ET模式,這種方式還有漏洞。

epoll的兩種模式LT和ET

二者的差異在于level-trigger模式下只要某個(gè)socket處于readable/writable狀態(tài),無論什么時(shí)候進(jìn)行epoll_wait都會(huì)返回該socket;而edge-trigger模式下只有某個(gè)socket從unreadable變?yōu)閞eadable或從unwritable變?yōu)閣ritable時(shí),epoll_wait才會(huì)返回該socket。如下兩個(gè)示意圖:

從socket讀數(shù)據(jù):

圖片

從socket寫數(shù)據(jù):

圖片

所以,在epoll的ET模式下,正確的讀寫方式為:

讀:只要可讀,就一直讀,直到返回0,或者 errno = EAGAIN 寫:只要可寫,就一直寫,直到數(shù)據(jù)發(fā)送完,或者 errno = EAGAIN。

正確的讀:

n = 0;
while ((nread = read(fd, buf + n, BUFSIZ-1)) > 0) {
if (nread == -1 && errno != EAGAIN)
{
perror("read error");
}
n += nread;
}

正確的寫:

int nwrite, data_size = strlen(buf);
n = data_size;
while (n > 0)
{
nwrite = write(fd, buf + data_size - n, n);
if (nwrite < n)
{
if (nwrite == -1 && errno != EAGAIN)
{
perror("write error");
}
break;
}
n -= nwrite;
}

accept上的問題

  • 阻塞模式 accept 存在的問題

考慮這種情況:TCP連接被客戶端夭折,即在服務(wù)器調(diào)用accept之前,客戶端主動(dòng)發(fā)送RST終止連接,導(dǎo)致剛剛建立的連接從就緒隊(duì)列中移出,如果套接口被設(shè)置成阻塞模式,服務(wù)器就會(huì)一直阻塞在accept調(diào)用上,直到其他某個(gè)客戶建立一個(gè)新的連接為止。但是在此期間,服務(wù)器單純地阻塞在accept調(diào)用上,就緒隊(duì)列中的其他描述符都得不到處理。

解決方案:把監(jiān)聽套接口設(shè)置為非阻塞,當(dāng)客戶在服務(wù)器調(diào)用accept之前中止某個(gè)連接時(shí),accept調(diào)用可以立即返回-1,這時(shí)源自Berkeley的實(shí)現(xiàn)會(huì)在內(nèi)核中處理該事件,并不會(huì)將該事件通知給epoll,而其他實(shí)現(xiàn)把errno設(shè)置為ECONNABORTED或者EPROTO錯(cuò)誤,我們應(yīng)該忽略這兩個(gè)錯(cuò)誤。

  • ET模式下accept存在的問題。

考慮這種情況:多個(gè)連接同時(shí)到達(dá),服務(wù)器的TCP就緒隊(duì)列瞬間積累多個(gè)就緒連接,由于是邊緣觸發(fā)模式,epoll只會(huì)通知一次,accept只處理一個(gè)連接,導(dǎo)致TCP就緒隊(duì)列中剩下的連接都得不到處理。

解決辦法是用while循環(huán)抱住accept調(diào)用,處理完TCP就緒隊(duì)列中的所有連接后再退出循環(huán)。如何知道是否處理完就緒隊(duì)列中的所有連接呢?accept返回-1并且errno設(shè)置為EAGAIN就表示所有連接都處理完。

綜合以上兩種情況,服務(wù)器應(yīng)該使用非阻塞地accept,accept在ET模式下的正確使用方式為:

while ((conn_sock = accept(listenfd,(struct sockaddr *) &remote, (size_t *)&addrlen)) > 0)
{
handle_client(conn_sock);
}
if (conn_sock == -1)
{
if (errno != EAGAIN && errno != ECONNABORTED && errno != EPROTO && errno != EINTR)
perror("accept");
}

一道騰訊后臺(tái)開發(fā)的面試題:

使用Linux epoll模型,水平觸發(fā)模式;當(dāng)socket可寫時(shí),會(huì)不停的觸發(fā)socket可寫的事件,如何處理?

第一種最普遍的方式:

需要向socket寫數(shù)據(jù)的時(shí)候才把socket加入epoll,等待可寫事件。接受到可寫事件后,調(diào)用write或者send發(fā)送數(shù)據(jù)。當(dāng)所有數(shù)據(jù)都寫完后,把socket移出epoll。

這種方式的缺點(diǎn)是,即使發(fā)送很少的數(shù)據(jù),也要把socket加入epoll,寫完后在移出epoll,有一定操作代價(jià)。

第二種的方式:

開始不把socket加入epoll,需要向socket寫數(shù)據(jù)的時(shí)候,直接調(diào)用write或者send發(fā)送數(shù)據(jù)。如果返回EAGAIN,把socket加入epoll,在epoll的驅(qū)動(dòng)下寫數(shù)據(jù),全部數(shù)據(jù)發(fā)送完畢后,再移出epoll。這種方式的優(yōu)點(diǎn)是:數(shù)據(jù)不多的時(shí)候可以避免epoll的事件處理,提高效率。

我自己代碼的問題

因?yàn)槲抑安庞玫氖欠亲枞鸈T模式,這樣我在發(fā)送緩沖數(shù)據(jù)的數(shù)據(jù),會(huì)出現(xiàn)EAGAIN的問題。這個(gè)問題并不可怕,最可怕的是會(huì)發(fā)生,因?yàn)樯厦嬉呀?jīng)有方法解決。為了看起來方便我這邊再拷貝一份。但是我源代碼采用的writev源代碼下載地址614行進(jìn)行發(fā)送,根本無法才采用下面的方法。

int nwrite, data_size = strlen(buf);
n = data_size;
while (n > 0)
{
nwrite = write(fd, buf + data_size - n, n);
if (nwrite < n)
{
if (nwrite == -1 && errno != EAGAIN)
{
perror("write error");
}
break;
}
n -= nwrite;
}

我的解決方法采用阻塞寫,這個(gè)方法很好的解決上面的問題:對(duì)于非阻塞的TCP套接字,如果緩沖區(qū)根本就沒空間,則返回一個(gè)EWOULDBLOCK錯(cuò)誤。如果緩沖區(qū)有一些空間,返回值是內(nèi)核能夠復(fù)制到該緩沖區(qū)的字節(jié)數(shù)。這個(gè)字節(jié)數(shù)也叫作不足計(jì)數(shù).。詳細(xì)見我上述連接的代碼文件。

在讀取緩沖區(qū)也是這樣。也是采用阻塞進(jìn)行讀取,這樣做,雖然降低并發(fā)性,但是為了準(zhǔn)確處理數(shù)據(jù)。

總結(jié)

ET模式下:

如果read返回0,那么說明已經(jīng)接受所有數(shù)據(jù) 如果errno=EAGAIN,說明還有數(shù)據(jù)未接收,等待下一次通知 如果read返回-1,說明發(fā)生錯(cuò)誤,停止處理

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 數(shù)據(jù)
    +關(guān)注

    關(guān)注

    8

    文章

    7256

    瀏覽量

    91836
  • 緩沖
    +關(guān)注

    關(guān)注

    0

    文章

    53

    瀏覽量

    18030
  • 線程
    +關(guān)注

    關(guān)注

    0

    文章

    508

    瀏覽量

    20208
  • epoll
    +關(guān)注

    關(guān)注

    0

    文章

    28

    瀏覽量

    3164
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    #硬聲創(chuàng)作季 網(wǎng)絡(luò)協(xié)議:多路復(fù)用EPOLL的面試題

    網(wǎng)絡(luò)協(xié)議復(fù)用多路復(fù)用多路復(fù)用網(wǎng)絡(luò)系統(tǒng)
    Mr_haohao
    發(fā)布于 :2022年10月16日 12:47:22

    [6.4.1]--多路復(fù)用

    多路復(fù)用數(shù)字邏輯
    李開鴻
    發(fā)布于 :2022年11月13日 01:18:45

    時(shí)鐘信號(hào)切換可以多路復(fù)用開關(guān)嗎

    FPGA設(shè)計(jì),外部時(shí)鐘有兩個(gè),切換時(shí)可以多路復(fù)用開關(guān)實(shí)現(xiàn)
    發(fā)表于 06-27 22:17

    如何在Mx1051的FlexCAN1中配置簡(jiǎn)單信號(hào)多路復(fù)用和擴(kuò)展信號(hào)多路復(fù)用

    我們正在研究 FlexCAN1 的 mxrt1051。我們是第一次在 FlexCAN 上工作,請(qǐng)協(xié)助以下幾點(diǎn): - 如何在 Mx1051 的 FlexCAN1 中配置簡(jiǎn)單信號(hào)多路復(fù)用和擴(kuò)展信號(hào)
    發(fā)表于 05-05 11:05

    多路復(fù)用與數(shù)字復(fù)接

    多路復(fù)用與數(shù)字復(fù)接8.1 頻分多路復(fù)用(FDM)原理8.2 時(shí)分多路復(fù)用(TDM)原理8.3 準(zhǔn)同步數(shù)字體系(PDH) 8.4 同步數(shù)字體系(SDH)  
    發(fā)表于 10-22 13:26 ?0次下載

    多路復(fù)用技術(shù)

    2.3  多路復(fù)用技術(shù)2.3.1  頻分多路復(fù)用2.3.2  時(shí)分多路復(fù)用2.3.3  波分多路復(fù)用2.3.4  碼分
    發(fā)表于 06-27 21:46 ?0次下載

    基于CPLD的非多路復(fù)用多路復(fù)用總線轉(zhuǎn)換橋的設(shè)計(jì)與實(shí)現(xiàn)

    基于CPLD的非多路復(fù)用多路復(fù)用總線轉(zhuǎn)換橋的設(shè)計(jì)與實(shí)現(xiàn) 微處理器對(duì)外并行總線接口方式一般分為兩種,一種為多路復(fù)用方式,數(shù)據(jù)與地址采用共用引腳,分時(shí)傳輸;另一
    發(fā)表于 03-28 15:08 ?942次閱讀
    基于CPLD的非<b class='flag-5'>多路復(fù)用</b>與<b class='flag-5'>多路復(fù)用</b>總線轉(zhuǎn)換橋的設(shè)計(jì)與<b class='flag-5'>實(shí)現(xiàn)</b>

    多路復(fù)用多路復(fù)用總線轉(zhuǎn)換橋的設(shè)計(jì)與實(shí)現(xiàn)

    多路復(fù)用多路復(fù)用總線轉(zhuǎn)換橋的設(shè)計(jì)與實(shí)現(xiàn) 提出了一種新穎的非多路復(fù)用總線與多路復(fù)用總線的轉(zhuǎn)換接口電路。以兩種總線的典型代表芯片TMS
    發(fā)表于 03-28 15:14 ?1069次閱讀
    非<b class='flag-5'>多路復(fù)用</b>與<b class='flag-5'>多路復(fù)用</b>總線轉(zhuǎn)換橋的設(shè)計(jì)與<b class='flag-5'>實(shí)現(xiàn)</b>

    多路復(fù)用多路復(fù)用總線轉(zhuǎn)換橋的設(shè)計(jì)與實(shí)現(xiàn)

    摘要:提出了一種新穎的非多路復(fù)用總線與多路復(fù)用總線的轉(zhuǎn)換接口電路。以兩種總線的典型代表芯片TMS320F206與SJA1000為例,分
    發(fā)表于 06-20 13:20 ?902次閱讀
    非<b class='flag-5'>多路復(fù)用</b>與<b class='flag-5'>多路復(fù)用</b>總線轉(zhuǎn)換橋的設(shè)計(jì)與<b class='flag-5'>實(shí)現(xiàn)</b>

    復(fù)用器的多路復(fù)用

    復(fù)用器的多路復(fù)用  多路復(fù)用
    發(fā)表于 01-07 14:27 ?1287次閱讀

    時(shí)分多路復(fù)用(TDM),時(shí)分多路復(fù)用(TDM)的原理是什么?

    時(shí)分多路復(fù)用(TDM),時(shí)分多路復(fù)用(TDM)的原理是什么?  為了提高信道利用率,使多個(gè)信號(hào)沿同一信道傳輸而互相不干擾,稱
    發(fā)表于 03-19 14:07 ?1.1w次閱讀

    什么是異步時(shí)分多路復(fù)用(ATDM)

    什么是異步時(shí)分多路復(fù)用(ATDM) 異步時(shí)分多路復(fù)用技術(shù) (ATDM,Asynchronism Time-Division Multiplexing)
    發(fā)表于 04-03 15:25 ?2061次閱讀

    時(shí)分多路復(fù)用(TDM),時(shí)分多路復(fù)用(TDM)是什么意思

    時(shí)分多路復(fù)用(TDM),時(shí)分多路復(fù)用(TDM)是什么意思 這種方法是把傳輸信道按時(shí)間分割,為每個(gè)用戶指定一個(gè)時(shí)間間隔,每個(gè)間隔里傳輸信號(hào)
    發(fā)表于 04-03 15:28 ?6346次閱讀

    IO多路復(fù)用基本概念

    一、IO多路復(fù)用基本概念 select、poll、epoll都是IO多路復(fù)用的機(jī)制。IO多路復(fù)用就是通過一種機(jī)制,讓一個(gè)進(jìn)程/線程可以監(jiān)視多個(gè)描述符,一旦某個(gè)描述符就緒(一般是讀寫就緒
    的頭像 發(fā)表于 11-10 16:34 ?1893次閱讀
    IO<b class='flag-5'>多路復(fù)用</b>基本概念

    頻分多路復(fù)用和時(shí)分多路復(fù)用的區(qū)別有哪些

    頻分多路復(fù)用(FDM)和時(shí)分多路復(fù)用(TDM)是兩種主要的多路復(fù)用技術(shù),它們?cè)谕ㄐ畔到y(tǒng)中扮演著至關(guān)重要的角色。
    的頭像 發(fā)表于 05-07 15:24 ?4792次閱讀