深入理解Nginx——事件

在 Nginx中,每一个事件都由 ngx_event_t结构体来表示。我们来了解ngx_event_t中每一个成员的含义,如下所示。

//在 Nginx 中,每一个事件都由ngx_event_t结构体来表示。
struct ngx_event_s {
    /*事件相关的对象。通常 data都是指向ngx_connectiont连接对象。开启文件异步I/O 时,它可
    能会指向 ngx_event_aio_t 结构体*/
    void            *data;
    /*标志位,为 1时表示事件是可写的。通常情况下,它表示对应的TCP连接目前状态是可写的,
    也就是连接处于可以发送网络包的状态*/
    unsigned         write:1;
    /*标志位,为1时表示为此事件可以建立新的连接。通常情况下,在ngx_cycle_t中的listening动
    态数组中,每一个监听对象ngx_listening_t 对应的读事件中的accept 标志位才会是1*/
    unsigned         accept:1;

    /* 这个标志位用于区分当前事件是否是过期的,它仅仅是给事件驱动模块使用的,
    而事件消费模块可不用关心。为什么需要这个标志位呢? 当开始处理一批事件时,
    处理前面的事件可能会关闭一些连接,而这些连接有可能影响这批事件中还未处理到的后面的事件。
    这时,可通过instance 标志位来避免处理后面的已经过期的事件。这是一个巧妙的设计方法*/

    /* used to detect the stale events in kqueue, rtsig, and epoll */
    unsigned         instance:1;

    /* 标志位,为 1 时表示当前事件是活跃的,为0时表示事件是不活跃的。
    这个状态对应着事件驱动模块处理方式的不同。例如,在添加事件、删除事件和处理事件时,
    actiye 标志位的不同都会对应着不同的处理方式。在使用事件时,一般不会直接改变 active 标志位*/
    /*
     * the event was passed or would be passed to a kernel;
     * in aio mode - operation was posted.
     */
    unsigned         active:1;
    /*标志位,为1时表示禁用事件,仅在 kqueue 或者rtsiq事件驱动模块中有效,
    而对于epoll事件驱动模块则无意义,这里不再详述*/

    unsigned         disabled:1;

    /*标志位,为1时表示当前事件已经准备就绪,也就是说,允许这个事件的消费模块处理这个事件。
    在HTTP 框架中,经常会检查事件的ready 标志位以确定是否可以接收请求或者发送响应*/

    /* the ready event; in aio mode 0 means that no operation can be posted */
    unsigned         ready:1;

    /* 该标志位仅对kgueue,eventport 等模块有意义,而对于Linux上的epoll事件驱动模块则是无意义的*/
    unsigned         oneshot:1;

    // 该标志位用于异步 AIO 事件的处理,
    /* aio operation is complete */
    unsigned         complete:1;

    //标志位,为1时表示当前处理的字符流已经结束
    unsigned         eof:1;
    //标志位,为1时表示事件在处理过程中出现错误
    unsigned         error:1;

    /*标志位,为1 时表示这个事件已经超时,用以提示事件的消费模块做超时处理*/
    unsigned         timedout:1;
    //标志位,为 1时表示这个事件存在于定时器中
    unsigned         timer_set:1;

    //标志位,delayed为1时表示需要延迟处理这个事件,它仅用于限速功能
    unsigned         delayed:1;

    //该标志位目前没有使用
    unsigned         read_discarded:1;
    //标志位,目前这个标志位未被使用
    unsigned         unexpected_eof:1;

    /*标志位,为 1时表示延迟建立TCP连接,也就是说,经过TCP三次握手后并不建立连接,
    而是要等到真正收到数据包后才会建立 TCP连接*/
    unsigned         deferred_accept:1;

    /*标志位,为 1时表示等待字符流结束,它只与 kqueue 和aio 事件驱动机制有关,不再详述*/
    /* the pending eof reported by kqueue or in aio chain operation */
    unsigned         pending_eof:1;

#if !(NGX_THREADS)
    //标志位,如果为 1,则表示在处理post事件时,当前事件已经准备就绪
    unsigned         posted_ready:1;
#endif

#if (NGX_WIN32)
    /* setsockopt(SO_UPDATE_ACCEPT_CONTEXT) was successful */
    unsigned         accept_context_updated:1;
#endif

#if (NGX_HAVE_KQUEUE)
    unsigned         kq_vnode:1;

    /* the pending errno reported by kqueue */
    int              kq_errno;
#endif

    /*
     * kqueue only:
     *   accept:     number of sockets that wait to be accepted
     *   read:       bytes to read when event is ready
     *               or lowat when event is set with NGX_LOWAT_EVENT flag
     *   write:      available space in buffer when event is ready
     *               or lowat when event is set with NGX_LOWAT_EVENT flag
     *
     * iocp: TODO
     *
     * otherwise:
     *   accept:     1 if accept many, 0 otherwise
     */

#if (NGX_HAVE_KQUEUE) || (NGX_HAVE_IOCP)
    int              available;
#else
    /*标志位,在epoll事件驱动机制下表示一次尽可能多地建立TCP连接,
    它与multi_accept 配置项对应 */
    unsigned         available:1;
#endif

    //这个事件发生时的处理方法,每个事件消费模块都会重新实现它。事件最核心的部分
    ngx_event_handler_pt  handler;


#if (NGX_HAVE_AIO)

#if (NGX_HAVE_IOCP)
    //Windows 系统下的一种事件驱动模型,这里不再详述
    ngx_event_ovlp_t ovlp;
#else
    //Linux aio机制中定义的结构体
    struct aiocb     aiocb;
#endif

#endif
    //由于 epoll 事件驱动方式不使用index,所以这里不再说明
    ngx_uint_t       index;

    //可用于记录error_log日志的ngx_log_t对象
    ngx_log_t       *log;

    // 定时器节点,用于定时器红黑树中
    ngx_rbtree_node_t   timer;

    // 标志位,为1时表示当前事件已经关闭,epoll模块没有使用它
    unsigned         closed:1;

    //该标志位日前无实际意义
    /* to test on worker exit */
    unsigned         channel:1;
    //该标志位日前无实际意义
    unsigned         resolver:1;

#if (NGX_THREADS)

    unsigned         locked:1;

    unsigned         posted_ready:1;
    unsigned         posted_timedout:1;
    unsigned         posted_eof:1;

#if (NGX_HAVE_KQUEUE)
    /* the pending errno reported by kqueue */
    int              posted_errno;
#endif

#if (NGX_HAVE_KQUEUE) || (NGX_HAVE_IOCP)
    int              posted_available;
#else
    unsigned         posted_available:1;
#endif

    ngx_atomic_t    *lock;
    ngx_atomic_t    *own_lock;

#endif

    /*post 事件将会构成一个队列再统一处理,这个队列以 next和prev 作为链表指针,
    以此构成一个简易的双向链表,其中 next 指向后一个事件的地址,prev 指向前一个事件的地址*/
    /* the links of the posted queue */
    ngx_event_t     *next;
    ngx_event_t    **prev;


#if 0

    /* the threads support */

    /*
     * the event thread context, we store it here
     * if $(CC) does not understand __thread declaration
     * and pthread_getspecific() is too costly
     */

    void            *thr_ctx;

#if (NGX_EVENT_T_PADDING)

    /* event should not cross cache line in SMP */

    uint32_t         padding[NGX_EVENT_T_PADDING];
#endif
#endif
};

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

三月微风

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值