libevent(九)evhttp
用libevent构建一个http server非常方便,可参考libevent(六)http server。
主要涉及的一个结构体是evhttp:
struct evhttp {
    /* Next vhost, if this is a vhost. */
    TAILQ_ENTRY(evhttp) next_vhost;
    /* All listeners for this host */
    TAILQ_HEAD(boundq, evhttp_bound_socket) sockets;
    TAILQ_HEAD(httpcbq, evhttp_cb) callbacks;
    /* All live connections on this host. */
    struct evconq connections;
    TAILQ_HEAD(vhostsq, evhttp) virtualhosts;
    TAILQ_HEAD(aliasq, evhttp_server_alias) aliases;
    /* NULL if this server is not a vhost */
    char *vhost_pattern;
    int timeout;
    size_t default_max_headers_size;
    ev_uint64_t default_max_body_size;
    /* Bitmask of all HTTP methods that we accept and pass to user
     * callbacks. */
    ev_uint16_t allowed_methods;
    /* Fallback callback if all the other callbacks for this connection
       don't match. */
    void (*gencb)(struct evhttp_request *req, void *);
    void *gencbarg;
    struct event_base *base;
};
值得关注的有两个成员:
  callbacks,一个链表,存放用户定义的回调函数
  connections,一个链表,存放所有连接,每个连接对应一个evhttp_connection
evhttp_connection结构如下:
/* A client or server connection. */
struct evhttp_connection {
/* we use this tailq only if this connection was created for an http
* server */
TAILQ_ENTRY(evhttp_connection) next; evutil_socket_t fd;
struct bufferevent *bufev; struct event retry_ev; /* for retrying connects */ char *bind_address; /* address to use for binding the src */
u_short bind_port; /* local port for binding the src */ char *address; /* address to connect to */
u_short port; size_t max_headers_size;
ev_uint64_t max_body_size; int flags;
#define EVHTTP_CON_INCOMING 0x0001 /* only one request on it ever */
#define EVHTTP_CON_OUTGOING 0x0002 /* multiple requests possible */
#define EVHTTP_CON_CLOSEDETECT 0x0004 /* detecting if persistent close */ int timeout; /* timeout in seconds for events */
int retry_cnt; /* retry count */
int retry_max; /* maximum number of retries */ enum evhttp_connection_state state; /* for server connections, the http server they are connected with */
struct evhttp *http_server; TAILQ_HEAD(evcon_requestq, evhttp_request) requests; void (*cb)(struct evhttp_connection *, void *);
void *cb_arg; void (*closecb)(struct evhttp_connection *, void *);
void *closecb_arg; struct deferred_cb read_more_deferred_cb; struct event_base *base;
struct evdns_base *dns_base;
};
值得关注的成员有两个:
  bufev,对应一个bufferevent
  requests,一个链表,存放该连接上的所有请求,每个请求对应evhttp_request
evhttp_request结构如下:
struct evhttp_request {
#if defined(TAILQ_ENTRY)
    TAILQ_ENTRY(evhttp_request) next;
#else
struct {
    struct evhttp_request *tqe_next;
    struct evhttp_request **tqe_prev;
}       next;
#endif
    /* the connection object that this request belongs to */
    struct evhttp_connection *evcon;
    int flags;
/** The request obj owns the evhttp connection and needs to free it */
#define EVHTTP_REQ_OWN_CONNECTION    0x0001
/** Request was made via a proxy */
#define EVHTTP_PROXY_REQUEST        0x0002
/** The request object is owned by the user; the user must free it */
#define EVHTTP_USER_OWNED        0x0004
/** The request will be used again upstack; freeing must be deferred */
#define EVHTTP_REQ_DEFER_FREE        0x0008
/** The request should be freed upstack */
#define EVHTTP_REQ_NEEDS_FREE        0x0010
    struct evkeyvalq *input_headers;
    struct evkeyvalq *output_headers;
    /* address of the remote host and the port connection came from */
    char *remote_host;
    ev_uint16_t remote_port;
    /* cache of the hostname for evhttp_request_get_host */
    char *host_cache;
    enum evhttp_request_kind kind;
    enum evhttp_cmd_type type;
    size_t headers_size;
    size_t body_size;
    char *uri;            /* uri after HTTP request was parsed */
    struct evhttp_uri *uri_elems;    /* uri elements */
    char major;            /* HTTP Major number */
    char minor;            /* HTTP Minor number */
    int response_code;        /* HTTP Response code */
    char *response_code_line;    /* Readable response */
    struct evbuffer *input_buffer;    /* read data */
    ev_int64_t ntoread;
    unsigned chunked:,        /* a chunked request */
        userdone:;            /* the user has sent all data */
    struct evbuffer *output_buffer;    /* outgoing post or data */
    /* Callback */
    void (*cb)(struct evhttp_request *, void *);
    void *cb_arg;
    /*
     * Chunked data callback - call for each completed chunk if
     * specified.  If not specified, all the data is delivered via
     * the regular callback.
     */
    void (*chunk_cb)(struct evhttp_request *, void *);
}; 
值得注意的是:
  每个请求有自己的输入缓冲input_buffer、输出缓冲output_buffer。
总结一下evhttp:
  1. 一个evhttp使用一个链表存放多个evhttp_connection,每个evhttp_connection使用链表存放多个evhttp_request。
  2. 每个evhttp_connection包含一个bufferevent,每个evhttp_request包含两个evbuffer,用于输入输出缓冲。
说了半天,好像没看见同步机制,可见evhttp不适合多线程。
参考资料:
libevent源码浅析: http库
libevent(九)evhttp的更多相关文章
- libevent(十三)evhttp事件处理流程
		在libevent(六)http server中,作为一个单线程http server,不仅要监听每个连接的到来,还要监听每个连接上的I/O事件. 查看源码可知,在evhttp_bind_socket ... 
- libevent(九)bufferevent
		bufferevent,带buffer的event struct bufferevent { struct event_base *ev_base; const struct bufferevent_ ... 
- 编译libevent源代码(Windows)
		学习笔记,只是记录本次成功用libevent源代码进行编译.环境为MinGW+VS2008+Msys. 0.下载libevent库 http://libevent.org/ 下载stable稳定版的库 ... 
- libevent(十)bufferevent 2
		接上文libevent(九)bufferevent 上文主要讲了bufferevent如何监听读事件,那么bufferevent如何监听写事件呢? 对于一个fd,只要它的写缓冲区没有满,就会触发写事件 ... 
- Thrift 基础(C++ rpc )
		一.thrift简介 thrift是Facebook开源的一套rpc框架,目前被许多公司使用 我理解的特点 使用IDL语言生成多语言的实现代码,程序员只需要实现自己的业务逻辑 支持序列化和反序列化操作 ... 
- libevhtp初探
		libevent的evhttp不适合多线程,libevhtp重新设计了libevent的http API,采用了和memcached类似的多线程模型. worker线程的管道读事件的回调函数为htp_ ... 
- libevent源码深度剖析九
		libevent源码深度剖析九 ——集成定时器事件 张亮 现在再来详细分析libevent中I/O事件和Timer事件的集成,与Signal相比,Timer事件的集成会直观和简单很多.Libevent ... 
- Libevent::evhttp服务器
		#include <cstdio> #include <stdio.h> #include <stdlib.h> #include <string.h> ... 
- Libevent::evhttp服务器下载
		void http_handler_Get_Download(struct evhttp_request *req, void *arg) { if (req == NULL) { return; } ... 
随机推荐
- 讲真,这两款idea插件,能治愈你英语不好的病
			时不时就有小伙伴问我,"二哥,能推荐一款 IDE 吗?"你看这话问的,现在搞 Java 的不都在用 Intellij IDEA 吗,还用得着推荐(我已经和 Eclipse 分手了) ... 
- beanshell自定义聚合报告时分线程组阶段展示
			假设现在一共会加载100个线程,期望聚合报告中分别展示1-20,20-40,40-60,60-80的四个阶段的线程并发性能数据,而不是总体的统计数据 beanshell脚本,具体内容: import ... 
- std::string::append函数
			string& append (const string& str); string& append (const string& str, size_t subpos ... 
- AJ学IOS 之CoreLocation基本使用
			猫猫分享,必须精品AJ 一:CoreLocation的基本使用 使用步骤: 首先导入头文件#import <CoreLocation/CoreLocation.h> 1.创建CoreLoc ... 
- Redis linux 下安装
			Redis linux 下安装 下载Redis安装包,可以从Redis中文网站中下载 下载地址:http://www.redis.cn/download.html Redis4.0 稳定版本 使用&l ... 
- 安卓menu的介绍与使用
			菜单之前是用户点击系统的菜单键才展示出来的,后来这个键渐渐被移除,菜单变成了点击任意的view都可以展示.菜单非为3种: 1.Options menu and action bar 选项菜单和操作栏 ... 
- OSI 七层模型以及TCP/IP模型
			OSI 七层模型 定义 OSI(Open System Interconnection)即开放式系统互联通信参考模型.该模型是国际标准化组织(ISO)制定的一个用于计算机或通信系统间互联的标准体系,一 ... 
- 《带你装B,带你飞》pytest修仙之路5 - yield操作
			1. 简介 上一篇中,我们刚刚实现了在每个用例之前执行初始化操作,那么用例执行完之后如需要清除数据(或还原)操作,可以使用 yield 来实现.fixture通过scope参数控制setup级别,既然 ... 
- Jmeter命令行执行并生成HTML报告
			前提:准备好jmeter脚本,找到jmeter配置文件查看生成的日志格式是否为csv,如果不是请改为csv 注意:使用命令执行jmeter脚本必须使用jmeter 3.0及以上版本1.使用命令行执行脚 ... 
- 空间小姐姐生活照,我用python破解加密压缩包,无忧查看
			事情的经过是这样的: 又是奶茶,行吧行吧. 快点开工,争取李大伟回来之前搞定. 李大伟说是6位数字密码 那么我们可以利用python生成全部的六位数字密码 #生成从000000到99999的密码表f ... 
