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

利用多線程和 C++ 實現一個簡單的 HTTP 服務器

開發 前端
本文介紹一種使用 C++ 和 多線程實現的簡單 HTTP 服務器。

前言:服務器是現代軟件不可或缺的一部分,而服務器的技術也是非常復雜和有趣的方向。隨著操作系統不斷地發展,服務器的底層架構也在不斷變化。本文介紹一種使用 C++ 和 多線程實現的簡單 HTTP 服務器。

首先我們先來看一下如何創建一個服務器。

int main() 
{
int server_fd;
struct sockaddr_in server_addr;
server_fd = socket(AF_INET, SOCK_STREAM, 0);
int on = 1;
setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
if (server_fd < 0) {
perror("create socket error");
goto EXIT;
}
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(8888);
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
if (bind(server_fd, (struct sockaddr*)&server_addr, sizeof(server_addr)) < 0) {
perror("bind address error");
goto EXIT;
}
if (listen(server_fd, 511) < 0) {
perror("listen port error");
goto EXIT;
}
while(1) {
int connfd = accept(server_fd, nullptr, nullptr);
if (connfd < 0)
{
perror("accept error");
continue;
}
// 處理
}
close(server_fd);
return 0;
EXIT:
exit(1);
}

我們看到根據操作系統提供的 API,創建一個 TCP 服務器非常簡單 ,只需要調用幾個函數就行。最后進程會阻塞在 accept 等待連接的到來,我們在一個死循環中串行地處理每個請求。顯然,這樣的效率肯定非常低,因為如果我們使用傳統的 read / write 函數的話,它是會引起進程阻塞的,這樣就會導致多個請求需要排隊進行處理。我們在此基礎上利用多線程提高一下效率。

std::thread threads[MAX_THREAD];
std::condition_variable condition_variable;
std::deque<int> requests;
std::mutex mutex;
for (int i = 0; i < MAX_THREAD; i++) {
threads[i] = std::thread(worker, &mutex, &condition_variable, &requests);
}

多線程就會涉及到并發 / 同步的問題,所以需要使用互斥變量和條件變量來處理這些問題。上面的代碼創建了幾個線程,然后在每個線程中執行 worker 函數來處理請求,除此之外,用 requests 變量來表示請求隊列,該變量會由主線程和子線程一起訪問。具體是由主線程生產任務,子線程消費。在了解子線程邏輯之前先看看主線程代碼的改動。

while(1) {
int connfd = accept(server_fd, nullptr, nullptr);
if (connfd < 0)
{
perror("accept error");
continue;
}
{
std::lock_guard<std::mutex> lock(mutex);
requests.push_back(connfd);
condition_variable.notify_one();
}
}

我們看到當主線程收到請求時,自己不處理,而是添加到請求隊列讓子線程處理,因為子線程沒有任務處理時會自我阻塞,所以主線程需要喚醒一個線程來處理新的請求。接下來看看子線程的邏輯。

void worker(std::mutex *mutex,
std::condition_variable *condition_variable,
std::deque<int> *requests) {
int connfd;
while (true) {
{
std::unique_lock<std::mutex> lock(*mutex);
// 沒有任務則等待,否則取出任務處理
while ((*requests).size() == 0)
{
(*condition_variable).wait(lock);
}
connfd = (*requests).front();
(*requests).pop_front();
}
char buf[4096];
int ret;
while (1) {
memset(buf, 0, sizeof(buf));
int bytes = read(connfd, buf, sizeof(buf));
if (bytes <= 0) {
close(connfd);
} else {
write(connfd, buf, bytes);
}
}
}
}

子線程不斷從任務隊列中取出任務,具體來說就是連接對應的文件描述符,然后不斷讀取里面的數據,最后返回給客戶端。但是這樣的功能顯然沒有太大意義,所以我們基于這個基礎上實現一個 HTTP 服務,讓它可以處理 HTTP 請求。當然我們手寫一個優秀的 HTTP 解析器并非易事,所以我們直接使用開源的就好,這里選擇的是 llhttp,這是 Node.js 所使用的 HTTP 解析器。這里就不具體羅列細節,大概介紹一下 llhttp 的用法。

typedef void (*p_on_headers_complete)(on_headers_complete_info, parser_callback);
typedef void (*p_on_body_complete)(on_body_complete_info, parser_callback);
typedef void (*p_on_body)(on_body_info, parser_callback);
struct parser_callback {
void * data;
p_on_headers_complete on_headers_complete;
p_on_body on_body;
p_on_body_complete on_body_complete;
};
class HTTP_Parser {
public:
HTTP_Parser(llhttp_type type, parser_callback callbacks = {});
int on_message_begin(llhttp_t* parser);
int on_status(llhttp_t* parser, const char* at, size_t length);
int on_url(llhttp_t* parser, const char* at, size_t length);
int on_header_field(llhttp_t* parser, const char* at, size_t length);
int on_header_value(llhttp_t* parser, const char* at, size_t length);
int on_headers_complete(llhttp_t* parser);
int on_body(llhttp_t* parser, const char* at, size_t length);
int on_message_complete(llhttp_t* parser);
int parse(const char* data, int len);
int finish();
void print();
};

HTTP_Parser 是我自己實現的 HTTP Parser Wrapper,主要是對 llhttp 的封裝,我們看到 HTTP_Parser 里有很多回調鉤子,對應的就是 llhttp 提供的,另外 HTTP_Parser 支持調用方傳入鉤子,也就是 parser_callback 所定義的。當 llhttp 回調 HTTP_Parser 時,HTTP_Parser 在合適的時機就會調用 parser_callback 里的回調,比如在解析完 HTTP Header 時,或者解析完整個報文時。具體的解析過程是當調用方收到數據時,執行 parse 函數,然后 llhttp 就會不斷地調用我們傳入的鉤子。了解了 HTTP 解析器的大致使用,我們來看看怎么在項目里使用。

parser_callback callback = {
&connfd,
[](on_body_complete_info info, parser_callback callback) {
int* connfd = (int *)callback.data;
const char * data = "HTTP/1.1 200 OK\r\nServer: multi-thread-server\r\ncontent-length: 11\r\n\r\nhello:world\r\n\r\n";
write(*connfd, data, strlen(data));
close(*connfd);
},
};
HTTP_Parser parser(HTTP_REQUEST, callback);
char buf[4096];
int ret;
while (1) {
memset(buf, 0, sizeof(buf));
int error = 0;
ret = read(connfd, buf, sizeof(buf));
parser.parse(buf, ret);
}

這里只列出關鍵的代碼,當我們收到數據時,我們通過 parser.parse(buf, ret) 調用 llhttp 進行解析,llhttp 就會不斷地回調鉤子函數,當解析完一個報文后,on_body_complete 回調就會被執行,在這里我們就可以對 HTTP 請求進行響應,比如這里返回一個 200 的響應報文,然后關閉連接。因為通過 llhttp 我們可以拿到具體的請求 url,所以我們還可以進一步拓展,根據 url 進行不同的處理。

到此為止,就實現了一個 HTTP 服務器了 ,在早期的時候,服務器也是采用這種多進程 / 多線程的處理方式,現在有了多路復用等技術后,很多服務器都是基于事件驅動來實現了。但是主線程接收請求,分發給子線程處理這種思想在有些服務器也還是存在的,比如 Node.js,只不過 Node.js 中是進程間進行傳遞。本文大概介紹到這里,服務器技術是非常復雜、有趣的方向,上層的架構也隨著操作系統的能力不斷在變化,本文只是作一個簡單的探索和興趣罷了,具體代碼在 https://github.com/theanarkh/multi-thread-server。下面是架構圖。

圖片

責任編輯:姜華 來源: 編程雜技
相關推薦

2019-04-24 15:06:37

Http服務器協議

2015-03-24 13:46:29

C++多線程計數器特性實現

2024-01-16 11:43:38

C++HashMap

2024-01-08 08:36:29

HTTPGo代理服務器

2009-02-27 11:15:00

多線程服務器MTS專用服務器

2018-03-01 10:45:25

HTTP服務器程序

2021-07-20 10:30:46

Golanghttp語言

2022-04-01 13:10:20

C++服務器代碼

2025-07-07 00:00:00

2019-05-08 14:37:49

Web服務器HTTP

2014-04-14 15:54:00

print()Web服務器

2017-12-27 09:49:35

HTTP服務器反向

2019-12-11 10:45:08

Python 開發編程語言

2016-11-08 18:53:08

編譯器

2009-08-25 01:46:00

C# WINDOWS服

2011-12-08 13:04:06

JavaNIO

2018-04-12 09:29:56

HTTP服務器問題

2018-06-15 10:25:43

Python HTTPFTP服務器

2018-12-06 09:23:33

2020-06-04 12:55:44

PyTorch分類器神經網絡
點贊
收藏

51CTO技術棧公眾號

黄色高清在线观看| 高清久久精品| 一卡二卡三卡日韩欧美| 成人毛片一区二区| 在线电影一区| 国产精品久久久久久av下载红粉| 日韩电影av| 亚洲第一级黄色片| 黄色片网站在线| 欧美性生活一区| 粉嫩tv在线播放| 亚洲二区在线视频| x88av蜜桃臀一区二区| 中文字幕佐山爱一区二区免费| 男人日女人下面视频| 国产精品综合在线视频| 先锋影音日韩| 久草这里只有精品视频| 亚洲国产一区二区精品视频| 一本久道久久久| 欧美人xxxxx| 免费在线成人网| 亚洲美女搞黄| 国产乱国产乱300精品| eeuss中文| 成人国产精品免费观看动漫| 久在线观看视频| 国产精品白丝在线| 国精产品999国精产品官网| 亚洲一区免费在线观看| 超碰在线电影| 欧美影片第一页| 污污在线观看| 日韩成人激情视频| 国产91亚洲精品久久久| 一区二区三区视频免费在线观看| 欧美成a人片免费观看久久五月天| 久久精品91久久久久久再现| 在线成人免费| 国产精品女主播| 韩国欧美一区| 欧美日韩视频免费在线观看| 国产毛片精品国产一区二区三区| 国产极品在线视频| 亚洲视频在线观看三级| 超碰在线中文| 欧美人成免费网站| 电影天堂国产精品| 欧美一级电影免费在线观看| 婷婷亚洲综合| 亚洲一卡二卡三卡| 中文字幕精品一区二区三区精品 | 欧美午夜一区二区三区| 成人免费高清| 日韩少妇与小伙激情| 婷婷综合一区| 国产自产在线视频一区| 韩国三级中文字幕hd久久精品| 5月婷婷6月丁香| 精品成人av一区| 超碰高清在线| 国产精品福利在线| 麻豆国产精品一区二区三区| 手机看片福利日韩| 欧美日韩国产区一| 精品国产乱码久久久久久樱花| 国产欧美欧洲在线观看| 久久午夜视频| 羞羞视频立即看| 欧美二区三区的天堂| 成人综合日日夜夜| 精品在线视频一区二区| 成人免费毛片嘿嘿连载视频| 宅男午夜电影| 亚洲最新av网址| 亚洲成人av| 国产特级黄色大片| 欧美精品电影在线播放| 亚洲精品观看| 亚洲精品无人区| 亚洲国产一二三| 成人国产精品| 久久99国产精品99久久| 中文字幕中文字幕一区| bl在线肉h视频大尺度| 国产成人亚洲精品| 国产99久久久国产精品潘金网站| 欧美变态视频| 久久精品这里热有精品| 日韩天天综合| 另类图片激情| 日韩亚洲在线观看| 视频一区免费在线观看| 一二三四中文在线| 久久久国产成人精品| 日韩高清不卡一区二区三区| 啊灬啊灬啊灬啊灬高潮在线看| 色999日韩欧美国产| 欧美亚洲一区二区三区| 蜜芽视频在线观看| 亚洲美女黄色片| 18成人在线观看| 久久这里有精品15一区二区三区| 久久久久99精品成人片| 色狮一区二区三区四区视频| 国产无限制自拍| 欧美日韩精品系列| 久久国产中文字幕| 国产日韩一区二区三区在线| 高h视频在线| 国产精品v欧美精品∨日韩| 精品一区二区三区免费| 国产在线中文字幕| 日韩免费精品视频| 国产色91在线| 亚洲精品三区| 激情成人开心网| 日韩丝袜美女视频| 一区二区高清| 高清av在线| 成人av片网址| 欧美午夜电影网| 99精品美女| 在线91av| 97人人模人人爽人人喊38tv| 亚洲一区二区偷拍精品| 欧美交a欧美精品喷水| 国产免费成人在线| 日韩中文字幕精品| 成人免费毛片片v| 不卡亚洲精品| 黄色影院一级片| 国产丝袜高跟一区| 精品一区二区三区久久| 欧美人与牲禽动交com| 茄子视频成人在线观看| 日韩一级精品视频在线观看| 国内一区二区三区| 老司机午夜在线视频| 九九九九久久久久| 日韩欧美的一区二区| 在线亚洲自拍| xxxx在线视频| www精品久久| 亚洲91av视频| 亚洲一二三区在线观看| 亚洲一区欧美| 手机av免费在线| 黄色大片中文字幕| 日本欧美一区二区在线观看| 首页综合国产亚洲丝袜| a'aaa级片在线观看| 最新二区三区av| 六月丁香婷婷在线| 日本一区二区三区视频在线观看| 国产精品欧美久久| 国产一区 二区 三区一级| 涩涩涩在线视频| 亚洲一区二区三区av无码| 日韩在线视频一区| 亚洲人成影院在线观看| 日韩av在线中文字幕| 欧美边添边摸边做边爱免费| 夜夜春亚洲嫩草影视日日摸夜夜添夜| 亚洲人成在线一二| 亚洲欧美在线观看| 午夜精品偷拍| h片在线观看下载| 五月婷婷狠狠操| 亚洲jizzjizz日本少妇| 日韩成人av网| 1024精品合集| 国产伦理一区| 日韩国产91| 中文资源在线网| 26uuu成人| 97视频在线免费观看| 欧美日韩成人综合天天影院| 国产一区中文字幕| 偷拍精品福利视频导航| 成人在线免费看黄| 激情视频免费网站| 好吊色欧美一区二区三区四区| 亚洲人成电影网站色| 1000精品久久久久久久久| 亚洲欧美日本视频在线观看| 9999精品| 免费**毛片在线| 色欲av无码一区二区人妻| 亚洲伊人久久大香线蕉av| 伊人久久久久久久久久| 欧美性猛交xxxx乱大交蜜桃| 国产成人亚洲精品青草天美| 精品一区二区三区在线| 桃子视频成人app| 美州a亚洲一视本频v色道| 成人免费无码av| 午夜视频久久久| 国产日韩在线观看av| 日韩一区二区三区在线播放|