日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪(fǎng)問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

基于Libevent的HTTP Server

發(fā)布時(shí)間:2023/12/2 编程问答 23 豆豆
生活随笔 收集整理的這篇文章主要介紹了 基于Libevent的HTTP Server 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.


簡(jiǎn)單的Http Server

使用Libevent內(nèi)置的http相關(guān)接口,可以很容易的構(gòu)建一個(gè)Http Server,一個(gè)簡(jiǎn)單的Http Server如下:

#include <event2/event.h> #include <event2/buffer.h> #include <event2/http.h> #include <Winsock2.h> #include <stdlib.h> #include <stdio.h>int init_win_socket() {WSADATA wsaData;if(WSAStartup(MAKEWORD(2,2) , &wsaData) != 0) {return -1;}return 0; }void generic_handler(struct evhttp_request *req, void *arg) {struct evbuffer *buf = evbuffer_new();if(!buf){puts("failed to create response buffer \n");return;}evbuffer_add_printf(buf, "Server Responsed. Requested: %s\n", evhttp_request_get_uri(req));evhttp_send_reply(req, HTTP_OK, "OK", buf);evbuffer_free(buf); }int main(int argc, char* argv[]) { #ifdef WIN32init_win_socket(); #endifshort http_port = 8081;char *http_addr = "127.0.0.1";struct event_base * base = event_base_new();struct evhttp * http_server = evhttp_new(base);if(!http_server){return -1;}int ret = evhttp_bind_socket(http_server,http_addr,http_port);if(ret!=0){return -1;}evhttp_set_gencb(http_server, generic_handler, NULL);printf("http server start OK! \n");event_base_dispatch(base);evhttp_free(http_server);WSACleanup();return 0; }

通過(guò)Libevent的接口構(gòu)建一個(gè)Http Server的過(guò)程如下:

(1)初始化:在event_base上新建一個(gè)evhttp,將這個(gè)evhttp綁定到監(jiān)聽(tīng)的IP和端口號(hào)。

(2)設(shè)置Http回調(diào)函數(shù):使用evhttp_set_gencb設(shè)置Http Server的處理請(qǐng)求的回調(diào)函數(shù)。

(3)啟動(dòng)Http Server:等待請(qǐng)求進(jìn)入事件循環(huán)。

在Http Server中使用定時(shí)器提供更新服務(wù)

#include <event2/event.h> #include <event2/buffer.h> #include <event2/http.h> #include <sys/stat.h> #include <Winsock2.h> #include <assert.h> #include <string.h> #include <stdlib.h> #include <stdio.h>#define DEFAULT_FILE "F:\\Libevent\\LibeventTest\\Debug\\sample.txt"char *filedata; time_t lasttime = 0; char filename[80]; int counter = 0;struct event *loadfile_event; struct timeval tv;void read_file() {unsigned long size = 0;char *data;struct stat buf;if(stat(filename,&buf)<0){printf("Read file error! \n");return;}if (buf.st_mtime > lasttime){if (counter++)fprintf(stderr,"Reloading file: %s",filename);elsefprintf(stderr,"Loading file: %s",filename);FILE *f = fopen(filename, "rb");if (f == NULL){fprintf(stderr,"Couldn't open file\n");return;}size = buf.st_size;filedata = (char *)malloc(size+1);memset(filedata,0,size+1);fread(filedata, sizeof(char), size, f);fclose(f);fprintf(stderr," (%d bytes)\n",size);lasttime = buf.st_mtime;} }void read_file_timer_cb(evutil_socket_t listener, short event, void *arg) {if (!evtimer_pending(loadfile_event, NULL)) {event_del(loadfile_event);evtimer_add(loadfile_event, &tv);}read_file(); }void load_file(struct event_base * base) {tv.tv_sec = 5;tv.tv_usec = 0;//loadfile_event = malloc(sizeof(struct event));loadfile_event = evtimer_new(base,read_file_timer_cb,NULL);//evtimer_set(loadfile_event,load_file,loadfile_event);evtimer_add(loadfile_event,&tv); }void generic_handler(struct evhttp_request *req, void *arg) {struct evbuffer *buf = evbuffer_new();if(!buf){puts("failed to create response buffer \n");return;}evbuffer_add_printf(buf,"%s",filedata);evhttp_send_reply(req, HTTP_OK, "OK", buf);evbuffer_free(buf); }int init_win_socket() {WSADATA wsaData;if(WSAStartup(MAKEWORD(2,2) , &wsaData) != 0) {return -1;}return 0; }int main(int argc, char* argv[]) { #ifdef WIN32init_win_socket(); #endifshort http_port = 8081;char *http_addr = "127.0.0.1";if (argc > 1){strcpy(filename,argv[1]);printf("Using %s\n",filename);}else{strcpy(filename,DEFAULT_FILE);}struct event_base * base = event_base_new();struct evhttp * http_server = evhttp_new(base);if(!http_server){return -1;}int ret = evhttp_bind_socket(http_server,http_addr,http_port);if(ret!=0){return -1;}evhttp_set_gencb(http_server, generic_handler, NULL);read_file();load_file(base);printf("http server start OK! \n");event_base_dispatch(base);evhttp_free(http_server);WSACleanup();return 0; }

在這個(gè)Http Server中提供了一個(gè)每5秒觸發(fā)一次的定時(shí)器,讀取一個(gè)文件,如果這個(gè)文件被更新過(guò),則讀取更新后的內(nèi)容。

當(dāng)訪(fǎng)問(wèn)這個(gè)Http Server時(shí),提供這個(gè)文件中最新的內(nèi)容。

多線(xiàn)程的Http Server

在上面的Http Server中,處理Http請(qǐng)求的回調(diào)函數(shù)generic_handler和定時(shí)器讀取文件的回調(diào)函數(shù)read_file_timer_cb都在同一個(gè)event_base的dispatch中,并且都在同一個(gè)進(jìn)程中,使用多線(xiàn)程可以改善程序的性能,下面是一個(gè)來(lái)自網(wǎng)絡(luò)的多線(xiàn)程Http Server:

#include <event.h> #include <evhttp.h> #include <pthread.h> #include <errno.h> #include <string.h> #include <fcntl.h> #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h>int httpserver_bindsocket(int port, int backlog); int httpserver_start(int port, int nthreads, int backlog); void* httpserver_Dispatch(void *arg); void httpserver_GenericHandler(struct evhttp_request *req, void *arg); void httpserver_ProcessRequest(struct evhttp_request *req);int httpserver_bindsocket(int port, int backlog) {int r;int nfd;nfd = socket(AF_INET, SOCK_STREAM, 0);if (nfd < 0) return -1;int one = 1;r = setsockopt(nfd, SOL_SOCKET, SO_REUSEADDR, (char *)&one, sizeof(int));struct sockaddr_in addr;memset(&addr, 0, sizeof(addr));addr.sin_family = AF_INET;addr.sin_addr.s_addr = INADDR_ANY;addr.sin_port = htons(port);r = bind(nfd, (struct sockaddr*)&addr, sizeof(addr));if (r < 0) return -1;r = listen(nfd, backlog);if (r < 0) return -1;int flags;if ((flags = fcntl(nfd, F_GETFL, 0)) < 0|| fcntl(nfd, F_SETFL, flags | O_NONBLOCK) < 0)return -1;return nfd; }int httpserver_start(int port, int nthreads, int backlog) {int r, i;int nfd = httpserver_bindsocket(port, backlog);if (nfd < 0) return -1;pthread_t ths[nthreads];for (i = 0; i < nthreads; i++) {struct event_base *base = event_init();if (base == NULL) return -1;struct evhttp *httpd = evhttp_new(base);if (httpd == NULL) return -1;r = evhttp_accept_socket(httpd, nfd);if (r != 0) return -1;evhttp_set_gencb(httpd, httpserver_GenericHandler, NULL);r = pthread_create(&ths[i], NULL, httpserver_Dispatch, base);if (r != 0) return -1;}for (i = 0; i < nthreads; i++) {pthread_join(ths[i], NULL);} }void* httpserver_Dispatch(void *arg) {event_base_dispatch((struct event_base*)arg);return NULL; }void httpserver_GenericHandler(struct evhttp_request *req, void *arg) {httpserver_ProcessRequest(req); }void httpserver_ProcessRequest(struct evhttp_request *req) {struct evbuffer *buf = evbuffer_new();if (buf == NULL) return;//here comes the magic }int main(void) {httpserver_start(80, 10, 10240); }

上面的代碼基于Libevent 1.X版本的,不過(guò)很容易很看懂:在一個(gè)監(jiān)聽(tīng)socket上創(chuàng)建了多個(gè)event_base實(shí)例和evhttp實(shí)例,在不同的線(xiàn)程中調(diào)度不同的event_base,繼而可以在不同的線(xiàn)程中處理http請(qǐng)求。

這里還有一個(gè)基于Libevent的多線(xiàn)程Http Server:https://sourceforge.net/projects/libevent-thread/,看源代碼處理的過(guò)程和上面類(lèi)似,只是每次在監(jiān)聽(tīng)的socket上accept一個(gè)連接請(qǐng)求時(shí),將對(duì)應(yīng)的處理放到一個(gè)工作隊(duì)列里,在隊(duì)列里由多線(xiàn)程處理相應(yīng)的回調(diào)函數(shù)。

?

總結(jié)

以上是生活随笔為你收集整理的基于Libevent的HTTP Server的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。