zoukankan      html  css  js  c++  java
  • (转)Libevent(4)— Bufferevent

    转自:http://name5566.com/4215.html

    参考文献列表:
    http://www.wangafu.net/~nickm/libevent-book/

    此文编写的时候,使用到的 Libevent 为 2.0.21

    Buffer IO 模式

    bufferevent 提供给我们一种 Buffer IO 模式(这里以写入数据为例):

    1. 在我们需要通过某个连接发送数据的时候,先将等待发送的数据放入到一个 buffer 中
    2. 等待此连接可以写入数据
    3. 尽可能多的获取 buffer 中的数据写入此连接
    4. 如果 buffer 中还有需要写入的数据则继续等待直到此连接可以写入数据

    每一个 bufferevent 都包含了一个输入 buffer 和一个输出 buffer,它们的类型为 evbuffer(结构体)。当我们向 bufferevent 写入数据的时候,实际上数据首先被写入到了输出 buffer,当 bufferevent 有数据可读时,我们实际上是从输入 buffer 中获取数据。

    目前 bufferevent 目前仅仅支持 stream-oriented 的协议(例如 TCP)并不支持 datagram-oriented 协议(例如 UDP)。一个 bufferevent 的实例负责一个特定的连接上的数据收发。

    Libevent 可以按需要创建多种类型的 bufferevent:

    1. 基于 socket 的 bufferevent。此类型的 bufferevent 使用 socket 来进行数据的收发,使用 event 机制来判断 socket 是否可以进行读写操作
    2. 异步 IO bufferevent。此类型的 bufferevent 使用 IOCP 接口实现(仅 Windows 下可用且目前处于实验阶段)
    3. Filtering bufferevent。此类型的 bufferevent 可以在同底层交互时完成一些额外的数据处理工作,例如可以完成数据的压缩和解析工作。这种类型的 bufferevent 的一个实例会封装了另外的一个 bufferevent,我们把这个被封装的 bufferevent 叫做底层 bufferevent
    4. Paired bufferevent。本文不谈

    bufferevent 的回调函数

    每个 bufferevent 实例可以有 3 个回调函数(通过接口 bufferevent_setcb 设置):

    1. 读取回调函数。默认情况下,只要从底层读取到了数据此回调函数将被调用
    2. 写入回调函数。默认情况下,足够多的数据被写入底层此回调函数将被调用
    3. 事件回调函数。当某些事件(错误)发生时被调用

    对于 buffer 的读、写和回调行为可以通过几个参数来配置,这几个参数在 Libevent 中被叫做 watermark(水位标记,我们可以将 buffer 想象为一个水池,水位标记用于标记水池中水的多少,也就是说,watermark 用于标记 buffer 中的数据量)。watermark 被实现为整数(类型为 size_t),有几种类型的 watermark:

    1. Read low-water mark 用于控制读取回调函数的行为。当 bufferevent 进行读取操作时,Read low-water mark 的值决定了输入 buffer 有多少数据后调用读取回调函数。默认的情况下,此值为 0,因此 bufferevent 读取操作都会导致读取回调函数被调用
    2. Read high-water mark 用于控制输入 buffer 的大小。如果输入 buffer 中的数据量达到 Read high-water mark 的值,那么 bufferevent 将停止读取。默认的情况下,此值为无限大
    3. Write low-water mark,用于控制写入回调函数的行为。当 bufferevent 进行写入操作时,Write low-water mark 的值决定了输出 buffer 有多少数据后调用写入回调函数。默认的情况下,此值为 0,因此写入回调函数会在输出 buffer 为空的时候被调用
    4. Write high-water mark,此值在使用 Filtering bufferevent 有特殊的用途

    在一些特殊需求中(详细并不讨论),我们可能需要回调函数被延时执行(这种被延时的回调函数被叫做 Deferred callbacks)。延时回调函数会在事件循环中排队,并在普通事件回调函数(regular event’s callback)之后被调用。

    从基于 socket 的 bufferevent 开始认识 bufferevent

    创建基于 socket 的 bufferevent:

    1. // 创建一个基于 socket 的 bufferevent
    2. // 函数执行失败返回 NULL
    3. struct bufferevent *bufferevent_socket_new(
    4. struct event_base *base,
    5. // socket 文件描述符
    6. // 此 socket 必须被设置为非阻塞的
    7. // 可以设置为 -1 表示之后再设置
    8. evutil_socket_t fd,
    9. // bufferevent 的选项
    10. enum bufferevent_options options);

    buffervent 的选项可以用来改变 bufferevent 的行为。可用的选项包括:

    1. BEV_OPT_CLOSE_ON_FREE
      当 bufferevent 被释放同时关闭底层(socket 被关闭等)
    2. BEV_OPT_THREADSAFE
      为 bufferevent 自动分配锁,这样能够在多线程环境中安全使用
    3. BEV_OPT_DEFER_CALLBACKS
      当设置了此标志,bufferevent 会延迟它的所有回调(参考前面说的延时回调)
    4. BEV_OPT_UNLOCK_CALLBACKS
      如果 bufferevent 被设置为线程安全的,用户提供的回调被调用时 bufferevent 的锁会被持有
      如果设置了此选项,Libevent 将在调用你的回调时释放 bufferevent 的锁

    建立连接:

    1. // address 和 addrlen 和标准的 connect 函数的参数没有区别
    2. // 如果 bufferevent bev 没有设置 socket(在创建时可以设置 socket)
    3. // 那么调用此函数将分配一个新的 socket 给 bev
    4. // 连接成功返回 0 失败返回 -1
    5. int bufferevent_socket_connect(struct bufferevent *bev,
    6. struct sockaddr *address, int addrlen);

    简单的一个范例:

    1. #include <event2/event.h>
    2. #include <event2/bufferevent.h>
    3. #include <sys/socket.h>
    4. #include <string.h>
    5.  
    6. // 事件回调函数
    7. void eventcb(struct bufferevent *bev, short events, void *ptr)
    8. {
    9. // 连接成功建立
    10. if (events & BEV_EVENT_CONNECTED) {
    11. /* We're connected to 127.0.0.1:8080. Ordinarily we'd do
    12. something here, like start reading or writing. */
    13. // 出现错误
    14. } else if (events & BEV_EVENT_ERROR) {
    15. /* An error occured while connecting. */
    16. }
    17. }
    18.  
    19. int main_loop(void)
    20. {
    21. struct event_base *base;
    22. struct bufferevent *bev;
    23. struct sockaddr_in sin;
    24.  
    25. base = event_base_new();
    26.  
    27. // 初始化连接地址
    28. memset(&sin, 0, sizeof(sin));
    29. sin.sin_family = AF_INET;
    30. sin.sin_addr.s_addr = htonl(0x7f000001); /* 127.0.0.1 */
    31. sin.sin_port = htons(8080); /* Port 8080 */
    32.  
    33. // 创建一个基于 socket 的 bufferevent
    34. // 参数 -1 表示并不为此 bufferevent 设置 socket
    35. bev = bufferevent_socket_new(base, -1, BEV_OPT_CLOSE_ON_FREE);
    36.  
    37. // 为 bufferevent bev 设置回调函数
    38. // 这里仅仅设置了事件回调函数
    39. // 后面会详细谈及此函数
    40. bufferevent_setcb(bev, NULL, NULL, eventcb, NULL);
    41.  
    42. // 进行连接
    43. if (bufferevent_socket_connect(bev,
    44. (struct sockaddr *)&sin, sizeof(sin)) < 0) {
    45. /* Error starting connection */
    46. bufferevent_free(bev);
    47. return -1;
    48. }
    49.  
    50. // 开始事件循环
    51. event_base_dispatch(base);
    52. return 0;
    53. }

    更多的 bufferevent API

    释放 bufferevent

    1. // 如果存在未完成的延时回调,bufferevent 会在回调完成后才被真正释放
    2. void bufferevent_free(struct bufferevent *bev);

    bufferevent 回调函数的设置和获取

    1. // 读取、写入回调函数原型
    2. // ctx 为用户自定义数据(由 bufferevent_setcb 设定)
    3. typedef void (*bufferevent_data_cb)(struct bufferevent *bev, void *ctx);
    4.  
    5. // 事件回调函数原型
    6. // ctx 为用户自定义数据(由 bufferevent_setcb 设定)
    7. // events 选项可以为:
    8. // BEV_EVENT_READING --- 在 bufferevent 上进行读取操作时出现了一个事件
    9. // BEV_EVENT_WRITING --- 在 bufferevent 上进行写入操作时出现了一个事件
    10. // BEV_EVENT_ERROR --- 进行 bufferevent 操作时(例如调用 bufferevent API)出错,获取详细的错误信息使用 EVUTIL_SOCKET_ERROR()
    11. // BEV_EVENT_TIMEOUT --- 在 bufferevent 上出现了超时
    12. // BEV_EVENT_EOF --- 在 bufferevent 上遇到了文件结束符
    13. // BEV_EVENT_CONNECTED --- 在 bufferevent 上请求连接完成了
    14. typedef void (*bufferevent_event_cb)(struct bufferevent *bev, short events, void *ctx);
    15.  
    16. // 设置回调函数
    17. // 如果希望禁用回调函数,那么设置对应的参数为 NULL
    18. void bufferevent_setcb(
    19. // bufferevent
    20. struct bufferevent *bufev,
    21. // 读取回调函数
    22. bufferevent_data_cb readcb,
    23. // 写入回调函数
    24. bufferevent_data_cb writecb,
    25. // 事件回调函数
    26. bufferevent_event_cb eventcb,
    27. // 用户定义的数据
    28. // 这三个回调函数均共享此参数
    29. void *cbarg
    30. );
    31.  
    32. // 取回回调函数
    33. // 参数为 NULL 表示忽略
    34. void bufferevent_getcb(
    35. struct bufferevent *bufev,
    36. bufferevent_data_cb *readcb_ptr,
    37. bufferevent_data_cb *writecb_ptr,
    38. bufferevent_event_cb *eventcb_ptr,
    39. void **cbarg_ptr
    40. );

    设置 watermark

    1. // events 参数可以为
    2. // EV_READ 表示设置 read watermark
    3. // EV_WRITE 表示设置 write watermark
    4. // EV_READ | EV_WRITE 表示设置 read 以及 write watermark
    5. void bufferevent_setwatermark(struct bufferevent *bufev, short events,
    6. size_t lowmark, size_t highmark);

    获取到输入和输出 buffer

    1. // 获取到输入 buffer
    2. struct evbuffer *bufferevent_get_input(struct bufferevent *bufev);
    3. // 获取到输出 buffer
    4. struct evbuffer *bufferevent_get_output(struct bufferevent *bufev);

    添加数据到输出 buffer

    1. // 下面两个函数执行成功返回 0 失败返回 -1
    2.  
    3. // 向 bufev 的输出 buffer 中添加大小为 size 的数据
    4. // 数据的首地址为 data
    5. int bufferevent_write(struct bufferevent *bufev,
    6. const void *data, size_t size);
    7. // 向 bufev 的输出 buffer 中添加数据
    8. // 数据来源于 buf
    9. // 此函数会清除 buf 中的所有数据
    10. int bufferevent_write_buffer(struct bufferevent *bufev,
    11. struct evbuffer *buf);

    从输入 buffer 中获取数据

    1. // 从 bufev 的输入 buffer 中获取最多 size 字节的数据保存在 data 指向的内存中
    2. // 此函数返回实际读取的字节数
    3. size_t bufferevent_read(struct bufferevent *bufev, void *data, size_t size);
    4. // 获取 bufev 的输入 buffer 中的所有数据并保存在 buf 中
    5. // 此函数成功返回 0 失败返回 -1
    6. int bufferevent_read_buffer(struct bufferevent *bufev,
    7. struct evbuffer *buf);

    关于 bufferevent 的一些高级话题,可以参考:http://www.wangafu.net/~nickm/libevent-book/Ref6a_advanced_bufferevents.html

    evbuffers

    evbuffer 是一个队列,在其尾部添加数据和在其头部删除数据均被优化了。evbuffer 相关的 API 在这里可以查看:http://www.wangafu.net/~nickm/libevent-book/Ref7_evbuffer.html

  • 相关阅读:
    【MyBatis】Inappropriate OGNL expression
    【java】前补零
    【js】前补零
    【Java】导出excel.xlsx
    【插件】fileinput
    【前端】WebSocket is already in CLOSING or CLOSED state?
    【HTML】input标签添加提示内容
    学习问题记录 -- 对象和引用
    八数码难题
    Java 逻辑运算符 & 与 &&的区别
  • 原文地址:https://www.cnblogs.com/buptlyn/p/4253499.html
Copyright © 2011-2022 走看看