国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

聊一聊 .NET在Linux下的IO多路復用select和epoll

系統 Linux
在windows平臺上,相信很多人都知道.NET異步機制是借助了Windows自帶的?IO完成端口?實現的異步交互,那在 Linux 下.NET 又是怎么玩的呢?主要還是傳統的 select,poll,epoll 的IO多路復用,在 coreclr源代碼中我們都能找到它們的影子。

一、背景

1. 講故事

在windows平臺上,相信很多人都知道.NET異步機制是借助了Windows自帶的 IO完成端口 實現的異步交互,那在 Linux 下.NET 又是怎么玩的呢?主要還是傳統的 select,poll,epoll 的IO多路復用,在 coreclr源代碼中我們都能找到它們的影子。

select & poll

在平臺適配層的 pal.cpp 文件中,有這樣的一句話。

#if HAVE_POLL
#include <poll.h>
#else
#include "pal/fakepoll.h"
#endif  // HAVE_POLL

簡而言之就是在不支持 poll 的linux版本中使用 select(fakepoll) 模擬,參考代碼如下:

圖片圖片

  2. epoll

同樣的在 linux 中你也會發現很多,截圖如下:

圖片圖片

二、select IO多路復用

1. select 解讀

在沒有 select 之前,我們需要手工管理多句柄的收發,在使用select IO多路復用技術之后,這些多句柄管理就由用戶轉交給linux系統了,這個也可以從核心的 select 函數看出。

int select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);
  • readfds,writefds,exceptfds

這三個字段依次監視著哪些句柄已成可讀狀態,哪些句柄已成可寫狀態,哪些句柄已成異常狀態,那技術上是如何實現的呢?在libc 中定義了一個 bit 數組,剛好文件句柄fd值作為 bit數組的索引,linux 在內核中只需要掃描 __fds_bits 中哪些位為1 即可找到需要監控的句柄。

/* fd_set for select and pselect.  */
typedef struct
  {
    /* XPG4.2 requires this member name.  Otherwise avoid the name
       from the global namespace.  */
#ifdef __USE_XOPEN
    __fd_mask fds_bits[__FD_SETSIZE / __NFDBITS];
# define __FDS_BITS(set) ((set)->fds_bits)
#else
    __fd_mask __fds_bits[__FD_SETSIZE / __NFDBITS];
# define __FDS_BITS(set) ((set)->__fds_bits)
#endif
  } fd_set;
  •  nfds,timeout

為了減少掃描范圍,提高程序性能,需要用戶指定一個最大的掃描值到 nfds 上。后面的timeout即超時時間。

2. select 的一個小例子

說了再多還不如一個例子有說服力,我們使用 select 機制對 Console 控制臺句柄 (STDIN_FILENO) 進行監控,一旦有數據進來立馬輸出,參考代碼如下:

#include <stdio.h>
#include <sys/select.h>
#include <unistd.h>

int main()
{
    fd_set readfds;
    struct timeval timeout;
    char buf[256];

    printf("Enter text (press Ctrl+D to end):\n");

    while (1)
    {
        FD_ZERO(&readfds);
        FD_SET(STDIN_FILENO, &readfds);
        timeout.tv_sec = 5; // 5秒超時
        timeout.tv_usec = 0;

        int ready = select(STDIN_FILENO + 1, &readfds, NULL, NULL, &timeout);

        if (ready == -1)
        {
            perror("select");
            break;
        }
        elseif (ready == 0)
        {
            printf("\nTimeout (5秒無輸入).\n");
            break;
        }
        elseif (FD_ISSET(STDIN_FILENO, &readfds))
        {
            // 使用 fgets 逐行讀取
            if (fgets(buf, sizeof(buf), stdin) != NULL)
            {
                printf("You entered: %s", buf); // 輸出整行(包含換行符)
            }
            else
            {
                printf("\nEnd of input (Ctrl+D pressed).\n");
                break;
            }
        }
    }

    return0;
}

圖片圖片

稍微解釋下代碼邏輯。

/* Standard file descriptors.  */
#define STDIN_FILENO 0 /* Standard input.  */
#define STDOUT_FILENO 1 /* Standard output.  */
#define STDERR_FILENO 2 /* Standard error output.  */
  • 將 STDIN_FILENO=0 塞入到可讀句柄監控 (readfds) 中。
  • 數據進來之后 select 被喚醒,執行后續邏輯。
  • 通過 FD_ISSET 判斷 bit=0 的位置(STDIN_FILENO)是否可用,可用的話讀取數據。

如果大家對 select 底層代碼感興趣,可以看下 linux 的 do_select 簡化實現,大量的遍歷邏輯(bit)。

static noinline_for_stack int do_select(int n, fd_set_bits *fds, struct timespec64 *end_time)
{
for (;;) {
unsignedlong *rinp, *routp, *rexp, *inp, *outp, *exp;
bool can_busy_loop = false;

  inp = fds->in; outp = fds->out; exp = fds->ex;
  rinp = fds->res_in; routp = fds->res_out; rexp = fds->res_ex;

for (i = 0; i < n; ++rinp, ++routp, ++rexp) {
   in = *inp++; out = *outp++; ex = *exp++;
   all_bits = in | out | ex;

   for (j = 0; j < BITS_PER_LONG; ++j, ++i, bit <<= 1) {
    mask = select_poll_one(i, wait, in, out, bit,busy_flag);
    if ((mask & POLLIN_SET) && (in & bit)) {
     res_in |= bit;
     retval++;
     wait->_qproc = NULL;
    }
    if ((mask & POLLOUT_SET) && (out & bit)) {
     res_out |= bit;
     retval++;
     wait->_qproc = NULL;
    }
    if ((mask & POLLEX_SET) && (ex & bit)) {
     res_ex |= bit;
     retval++;
     wait->_qproc = NULL;
    }
   }
  }

if (!poll_schedule_timeout(&table, TASK_INTERRUPTIBLE, to, slack))
   timed_out = 1;
 }

return retval;
}

三、epoll IO多路復用

1. epoll 解讀

現在主流的軟件(Redis,Nigix) 都是采用 epoll,它解決了select低效的遍歷,畢竟數組最多支持1024個bit位,一旦句柄過多會影響異步讀取的效率。epoll的底層借助了。

  • 紅黑樹:對句柄進行管理,復雜度為 O(logN)。
  • 就緒隊列:一旦句柄變得可讀或可寫,內核會直接將句柄送到就緒隊列。

libc中使用 epoll_wait 函數監視著就緒隊列,一旦有數據立即提取,復雜度 O(1),其實這個機制和 Windows 的IO完成端口 已經很靠近了,最后配一下參考代碼。

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/epoll.h>
#include <fcntl.h>
#include <errno.h>

#define MAX_EVENTS 10   // 最大監聽事件數
#define TIMEOUT_MS 5000 // epoll_wait 超時時間(毫秒)

int main()
{
    int epoll_fd, nfds;                        // epoll 文件描述符和返回的事件數
    struct epoll_event ev, events[MAX_EVENTS];// epoll 事件結構體
    char buf[256];

    // 創建 epoll 實例
    epoll_fd = epoll_create1(0);
    if (epoll_fd == -1)
    {
        perror("epoll_create1");
        exit(EXIT_FAILURE);
    }

    // 配置并添加標準輸入到 epoll 監聽
    ev.events = EPOLLIN;       // 監聽文件描述符的可讀事件(輸入)
    ev.data.fd = STDIN_FILENO; // 監聽標準輸入(文件描述符 0)

    if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, STDIN_FILENO, &ev) == -1)
    {
        perror("epoll_ctl: STDIN_FILENO");
        exit(EXIT_FAILURE);
    }

    printf("Enter text line by line (press Ctrl+D to end):\n");

    // 主循環:監聽事件
    while (1)
    {
        // 等待事件發生或超時
        nfds = epoll_wait(epoll_fd, events, MAX_EVENTS, TIMEOUT_MS);

        if (nfds == -1)
        {
            perror("epoll_wait");
            break;
        }
        elseif (nfds == 0)
        {
            printf("\nTimeout (5秒無輸入).\n");
            break;
        }

        // 處理所有觸發的事件
        for (int n = 0; n < nfds; ++n)
        {
            if (events[n].data.fd == STDIN_FILENO)
            {
                // 使用 fgets 逐行讀取輸入
                if (fgets(buf, sizeof(buf), stdin) != NULL)
                {
                    printf("You entered: %s", buf);
                }
                else
                {
                    // 輸入結束(用戶按下 Ctrl+D)
                    printf("\nEnd of input (Ctrl+D pressed).\n");
                    break;
                }
            }
        }
    }

    close(epoll_fd);
    return0;
}

圖片圖片

四、總結

說了這么多,文尾總結下目前主流的 epoll 和 iocp 各自的特點。

特性

epoll (Linux)

IOCP (Windows)

模型

事件驅動 (Reactor)

完成端口 (Proactor)

核心思想

通知可讀寫事件

通知I/O操作完成

適用場景

高并發網絡編程

高并發I/O操作

編程復雜度

較低

較高

網絡I/O性能

極佳(百萬級連接)

優秀

磁盤I/O支持

有限

完善

CPU利用率

內存開銷


責任編輯:武曉燕 來源: 一線碼農聊技術
相關推薦

2021-05-31 06:50:47

SelectPoll系統

2023-03-01 14:32:31

redisIOEpoll

2021-03-05 11:26:42

面試Java程序

2023-12-13 09:45:49

模型程序

2023-01-09 10:04:47

IO多路復用模型

2022-09-12 06:33:15

Select多路復用

2024-08-08 14:57:32

2023-12-06 07:16:31

Go語言語句

2020-10-14 09:11:44

IO 多路復用實現機

2021-07-16 11:48:26

模型 .NET微軟

2022-08-26 00:21:44

IO模型線程

2019-02-13 14:15:59

Linux版本Fedora

2020-06-28 09:30:37

Linux內存操作系統

2025-05-13 07:10:31

2025-04-24 10:05:51

2023-11-07 08:19:35

IO多路復用磁盤、

2021-08-26 09:31:40

Nacos配置注冊

2021-01-04 08:09:07

Linux內核Watchdog

2024-12-30 00:00:05

2021-01-26 05:06:24

LinuxXargs 命令
點贊
收藏

51CTO技術棧公眾號

久久久亚洲网站| 欧美午夜欧美| 白浆在线视频| 国产精品护士白丝一区av| 色播亚洲婷婷| 欧美激情视频一区二区三区在线播放 | 一区二区高清免费观看影视大全| 中文字幕中文字幕一区三区| 国产综合精品| 国产精品电影在线观看| 久久九九精品视频| 亚洲午夜精品久久久久久性色 | av大片在线观看| 亚洲综合色噜噜狠狠| 熟妇人妻va精品中文字幕| 国产精品2024| 亚洲女人天堂在线| 国产精品日本一区二区不卡视频| 啪啪亚洲精品| 福利一区在线| 亚洲欧美国产精品久久久久久久 | 在线观看网站免费入口在线观看国内 | 国内外成人免费视频| 中文字幕在线视频久| 热久久一区二区| 久草视频这里只有精品| 中文字幕在线不卡一区二区三区| 欧美成人黑人xx视频免费观看| h片在线观看视频免费| 91精品国产综合久久久蜜臀粉嫩 | 亚洲国产精品人人做人人爽| 国产激情二区| 亚洲一区二区三区免费视频| y4480在线8影院| 精品久久久久久久久中文字幕 | 精品一区二区精品| 永久免费精品视频网站| 蜜桃久久久久久久| 免费看日本黄色| 久久久高清一区二区三区| 色多多视频在线播放| 亚洲一区二区四区蜜桃| 成年网站在线| 亚洲成人xxx| 久久99久久久精品欧美| 国内精品久久影院| 仙踪林久久久久久久999| 九9re精品视频在线观看re6| 蜜桃视频在线观看一区| 免费在线a视频| 亚洲欧美一区二区久久| 青青草在线播放| 日韩精品中文字幕在线一区| 日本精品不卡| 91av在线播放| 亚洲一区免费| 777精品久无码人妻蜜桃| 亚洲精品一卡二卡| 最新97超碰在线| 在线亚洲欧美视频| 国产一区网站| 亚洲巨乳在线观看| 中文字幕第一区二区| 免费在线稳定资源站| 亚洲精品永久免费| 国产日产精品_国产精品毛片| 高清视频一区| 97久久精品人人做人人爽| 在线观看的av网站| 亚洲性夜色噜噜噜7777| 欧美精品尤物在线观看| 亚洲国产午夜伦理片大全在线观看网站 | 国产精品丝袜久久久久久消防器材| 一色屋精品亚洲香蕉网站| 欧美色综合一区二区三区| 亚洲一区www| 婷婷综合视频| 国产精品国产亚洲精品看不卡| 精品久久久久久久久久久| 欧美magnet| 亚洲aⅴ男人的天堂在线观看| 国产999精品久久久久久| 国产传媒在线视频| 亚洲视频在线播放| 99久久夜色精品国产亚洲96| 国产在线视频综合| 欧美天天综合网| 国产精东传媒成人av电影| 手机在线观看国产精品| 亚洲国产一二三| 成人豆花视频| 欧美亚洲精品日韩| 亚洲欧美日韩系列| 性爽视频在线| 动漫精品视频| 亚洲欧美色一区| 日韩一区二区三区在线免费观看 | 巨大荫蒂视频欧美大片| 欧美激情久久久久久| 久久精品盗摄| 99热一区二区| 亚洲色图色老头| 最新国产拍偷乱拍精品| 久久久久久77777| 综合国产在线观看| 奇米影视一区二区三区小说| 涩爱av在线播放一区二区| 欧美激情在线视频二区| 国产成人亚洲精品青草天美| 超碰人人在线| 99re国产视频| 亚洲欧美日韩国产综合| 91麻豆精品国产综合久久久| 欧美日韩亚洲免费| 欧美性猛交xxxx偷拍洗澡| 台湾佬综合网| av网站在线不卡| 日韩在线一区二区三区免费视频| 日精品一区二区| 国产私拍精品| 欧洲成人在线观看| 国产午夜精品久久| 国产一区二区三区免费观看在线| 国产精品一区在线免费观看| 4hu四虎永久在线影院成人| 欧美喷水视频| 久久免费看视频| 国产日本欧美一区二区三区| 亚洲三级久久久| 18国产精品| 中文字幕第80页| 欧美丰满少妇xxxxx做受| 97久久精品人人澡人人爽| 国产精品麻豆成人av电影艾秋| 波多野结衣 作品| 在线成人激情视频| 粉嫩av一区二区三区粉嫩 | 大伊香蕉精品在线品播放| 青青青国产在线观看| 亚洲男人天堂手机在线| 国内精品免费在线观看| 国产欧美一区二区三区精品酒店| 亚洲国产一区二区三区在线| 亚洲精品一区二区三区蜜桃下载| 久久久人人人| av中文字幕电影在线看| 中文字幕一区二区三区在线乱码 | 亚洲性生活视频| eeuss鲁片一区二区三区在线观看| 精品3atv在线视频| 国内自拍在线观看| 久久久久久中文| 国产精品福利一区| 欧美偷拍综合| 国产高清视频免费最新在线| 欧美日韩国产综合在线| 亚洲人成网站999久久久综合| 成人av免费在线| 99国产精品免费网站| 免费成人高清| 国产精品免费看一区二区三区| 欧美狂野另类xxxxoooo| 久久国产三级精品| 亚瑟国产精品| xxxx影院| 精品人伦一区二区三区| 亚洲老司机av| 欧美经典三级视频一区二区三区| 成人直播大秀| 欧美一区二区三区在线观看免费| 亚洲午夜精品久久| 欧美日本精品在线| 午夜电影久久久| 日韩电影一区二区三区四区| 日本免费成人| 水莓100国产免费av在线播放| 婷婷五月色综合| 欧美精品久久久久久久久| 欧美三级免费观看| 国产成人日日夜夜| 欧美亚洲激情| www.成人爱| 免费黄色av电影| 亚洲一区二区精品在线观看| 久久久久久97| 91麻豆精品国产| 欧美国产日韩亚洲一区| 亚洲国产激情| 51社区在线成人免费视频| 99免在线观看免费视频高清| 欧美变态另类刺激| 国产欧美综合精品一区二区| 欧美xxxx做受欧美.88| 欧美日韩国产一二三| 久久久一区二区| 亚洲一区二区三区高清| 深夜福利一区二区三区| 亚洲成a人片在线不卡一二三区| 国产亚洲精品v| 大香伊人久久精品一区二区|