基于Libevent的HTTP Server

时间:2022-09-03 14:08:08

简单的Http Server

使用Libevent内置的http相关接口,可以很容易的构建一个Http Server,一个简单的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(,) , &wsaData) != )
{
return -;
}
return ;
} 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 WIN32
init_win_socket();
#endif short http_port = ;
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 -;
} int ret = evhttp_bind_socket(http_server,http_addr,http_port);
if(ret!=)
{
return -;
} evhttp_set_gencb(http_server, generic_handler, NULL); printf("http server start OK! \n"); event_base_dispatch(base); evhttp_free(http_server); WSACleanup();
return ;
}

通过Libevent的接口构建一个Http Server的过程如下:

(1)初始化:在event_base上新建一个evhttp,将这个evhttp绑定到监听的IP和端口号。

(2)设置Http回调函数:使用evhttp_set_gencb设置Http Server的处理请求的回调函数。

(3)启动Http Server:等待请求进入事件循环。

在Http Server中使用定时器提供更新服务

#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 = ;
char filename[];
int counter = ; struct event *loadfile_event;
struct timeval tv; void read_file()
{
unsigned long size = ;
char *data;
struct stat buf; if(stat(filename,&buf)<)
{
printf("Read file error! \n");
return;
} if (buf.st_mtime > lasttime)
{
if (counter++)
fprintf(stderr,"Reloading file: %s",filename);
else
fprintf(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+);
memset(filedata,,size+);
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 = ;
tv.tv_usec = ; //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(,) , &wsaData) != )
{
return -;
}
return ;
} int main(int argc, char* argv[])
{
#ifdef WIN32
init_win_socket();
#endif short http_port = ;
char *http_addr = "127.0.0.1"; if (argc > )
{
strcpy(filename,argv[]);
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 -;
} int ret = evhttp_bind_socket(http_server,http_addr,http_port);
if(ret!=)
{
return -;
} 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 ;
}

在这个Http Server中提供了一个每5秒触发一次的定时器,读取一个文件,如果这个文件被更新过,则读取更新后的内容。

当访问这个Http Server时,提供这个文件中最新的内容。

多线程的Http Server

在上面的Http Server中,处理Http请求的回调函数generic_handler和定时器读取文件的回调函数read_file_timer_cb都在同一个event_base的dispatch中,并且都在同一个进程中,使用多线程可以改善程序的性能,下面是一个来自网络的多线程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, );
if (nfd < ) return -; int one = ;
r = setsockopt(nfd, SOL_SOCKET, SO_REUSEADDR, (char *)&one, sizeof(int)); struct sockaddr_in addr;
memset(&addr, , 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 < ) return -;
r = listen(nfd, backlog);
if (r < ) return -; int flags;
if ((flags = fcntl(nfd, F_GETFL, )) <
|| fcntl(nfd, F_SETFL, flags | O_NONBLOCK) < )
return -; return nfd;
} int httpserver_start(int port, int nthreads, int backlog) {
int r, i;
int nfd = httpserver_bindsocket(port, backlog);
if (nfd < ) return -;
pthread_t ths[nthreads];
for (i = ; i < nthreads; i++) {
struct event_base *base = event_init();
if (base == NULL) return -;
struct evhttp *httpd = evhttp_new(base);
if (httpd == NULL) return -;
r = evhttp_accept_socket(httpd, nfd);
if (r != ) return -;
evhttp_set_gencb(httpd, httpserver_GenericHandler, NULL);
r = pthread_create(&ths[i], NULL, httpserver_Dispatch, base);
if (r != ) return -;
}
for (i = ; 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(, , );
}

上面的代码基于Libevent 1.X版本的,不过很容易很看懂:在一个监听socket上创建了多个event_base实例和evhttp实例,在不同的线程中调度不同的event_base,继而可以在不同的线程中处理http请求。

这里还有一个基于Libevent的多线程Http Server:https://sourceforge.net/projects/libevent-thread/,看源代码处理的过程和上面类似,只是每次在监听的socket上accept一个连接请求时,将对应的处理放到一个工作队列里,在队列里由多线程处理相应的回调函数。

基于Libevent的HTTP Server的更多相关文章

  1. PHP写的异步高并发服务器,基于libevent

    PHP写的异步高并发服务器,基于libevent 博客分类: PHP PHPFPSocketLinuxQQ  本文章于2013年11月修改. swoole已使用C重写作为PHP扩展来运行.项目地址:h ...

  2. 基于libevent的tcp拆包分包库

    TCP/IP协议虽然方便,但是由于是基于流的传输(UDP是基于数据报的传输),无论什么项目,总少不了解决拆包分包问题. 以前的项目总是每个程序员自己写一套拆包分包逻辑,实现的方法与稳定性都不太一致.终 ...

  3. 基于 libevent 开发的 C&plus;&plus; 11 高性能网络服务器 evpp(360的作品)

    evpp是一个基于libevent开发的现代化C++11高性能网络服务器,自带TCP/UDP/HTTP等协议的异步非阻塞式的服务器和客户端库. 特性: 现代版的C++11接口 非阻塞异步接口都是C++ ...

  4. 基于iSCSI的SQL Server 2012群集测试&lpar;四&rpar;--模拟群集故障转移

    6.模拟群集故障转移 6.1 模拟手动故障转移(1+1) 模拟手动故障转移的目的有以下几点: 测试群集是否能正常故障转移 测试修改端口是否能同步到备节点 测试禁用full-text和Browser服务 ...

  5. 怎样基于谷歌地图的Server缓存公布Image Service服务

    怎样基于谷歌地图的Server缓存公布Image Service服务 第一步:下载地图数据 下载安装水经注万能地图下载器,启动时仅仅选择电子.谷歌(这里能够依据自己的须要选择).例如以下图所看到的. ...

  6. 基于libevent&comma; libuv和android Looper不断演进socket编程 - 走向架构师之路 - 博客频道 - CSDN&period;NET

    基于libevent, libuv和android Looper不断演进socket编程 - 走向架构师之路 - 博客频道 - CSDN.NET 基于libevent, libuv和android L ...

  7. C&num; &period;net基于Http实现web server&lpar;web服务&rpar;

    原文:C# .net基于Http实现web server(web服务) 什么是 web server?  百度百科是这么解释的: Web Server中文名称叫网页服务器或web服务器.WEB服务器也 ...

  8. 基于moco的mock server 简单应用 来玩玩吧

    提起mock大家应该就知道是干嘛用的了,再次再介绍一种简单的方式,基于moco的mock server.步骤很简单: 1. 首先,要下载个moco的jar0_1482402640757_moco-ru ...

  9. 基于I&sol;O的Server&sol;Client实现

    在前面的文章中讲了基于NIO实现的Server/Client.本文就讲讲基于同步堵塞式I/O实现的Server/Client好与前面的NIO中的Server/Client进行对照. 网络编程中须要解决 ...

随机推荐

  1. C&num;&lowbar;技巧:窗口抖动

    原理 * 窗口抖动:即每隔一段很小的时间,窗口位置发生变化  * 时间控制:利用for循环||利用timer * 窗口位置发生变化:控件Left/Top属性或Location属性, 注:Left/To ...

  2. python学习笔记系列----(六)错误和异常

    python至少有2类不同的错误:语法错误(Syntax Errors)和异常(Exceptions). 8.1 语法错误 这个单词应该还是很有必要认识的,呵呵,语法错误,也叫解析错误,是我们最不愿意 ...

  3. js this理解

    原文链接:http://www.ruanyifeng.com/blog/2010/04/using_this_keyword_in_javascript.html this是js语言的几个关键字,代表 ...

  4. 支付宝APP支付开发- IOException &colon; DerInputStream&period;getLength&lpar;&rpar;&colon; lengthTag&equals;127&comma; too big&period;

    支付宝APP支付Java开发报错: IOException : DerInputStream.getLength(): lengthTag=127, too big. 后来排查是因为没有设置私钥.

  5. ASP&period;NET MVC3入门教程之第一个WEB应用程序

    本文转载自:http://www.youarebug.com/forum.php?mod=viewthread&tid=91&extra=page%3D1 上一节,我们已经搭建好了AS ...

  6. BZOJ3564 &colon; &lbrack;SHOI2014&rsqb;信号增幅仪

    先把所有点绕原点逆时针旋转(360-a)度,再把所有点横坐标除以放大倍数p,最后用随机增量法求最小圆覆盖即可. 时间复杂度期望$O(n)$ #include<cstdio> #includ ...

  7. a different object with the same identifier value was already associat

    问题:这个著名的托管态update更新异常 org.hibernate.NonUniqueObjectException: a different object with the same ident ...

  8. Solr高亮显示highlight的三种实现

    高亮显示在搜索中使用的比较多,比较常用的有三种使用方式,如果要对某field做高亮显示,必须对该field设置stored=true      第一种是普通的高亮显示Highlighter,根据查询的 ...

  9. C&num; 8中的范围类型&lpar;Range Type&rpar;

    C# 8.0中加入了一个新的范围类型(Range Type). 这里我们首先展示一些代码,并一步一步为代码添加一些不同的东西, 为大家展示一下范围类型的功能和用法. 我们最原始的代码如下: stati ...

  10. HDU 4311 Meeting point-1(曼哈顿距离最小)

    http://acm.hdu.edu.cn/showproblem.php?pid=4311 题意:在二维坐标中有n个点,现在要从这n个点中选出一个点,使得其他点到该点的曼哈顿距离总和最小. 思路: ...