nginx中的阻塞和非阻塞设置

探讨NGINX如何通过设置非阻塞模式避免线程阻塞,特别是在使用select和epoll模型时的做法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

纵然对于select,epoll这2个模型,既可以使用阻塞模式的socket也可以使用非阻塞的

虽然,但是这2个模型的消息通知可以防止accetp,recv在阻塞模式下的进入阻塞状态,却不能防止NGINX主动发出的connect和send进入阻塞状态


特别是NGINX是单线程模型,任何调用的阻塞状态都会造成整个线程阻塞甚至挂起,这对NGINX来说是完全不能接受的


因此在ngx_open_listening_sockets函数中,对所有listen socket,除非是IOCP或者UNIX_AIO模型,都把这个socket设置成非阻塞模式,代码如下:

            if (!(ngx_event_flags & NGX_USE_AIO_EVENT)) {
                if (ngx_nonblocking(s) == -1) {
                    ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno,
                                  ngx_nonblocking_n " %V failed",
                                  &ls[i].addr_text);


                    if (ngx_close_socket(s) == -1) {
                        ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno,
                                      ngx_close_socket_n " %V failed",
                                      &ls[i].addr_text);
                    }


                    return NGX_ERROR;
                }
            }

而在ngx_event_accept中,也有一样的实现逻辑

        /* set a blocking mode for aio and non-blocking mode for others */


        if (ngx_inherited_nonblocking) {
            if (ngx_event_flags & NGX_USE_AIO_EVENT) {
                if (ngx_blocking(s) == -1) {
                    ngx_log_error(NGX_LOG_ALERT, ev->log, ngx_socket_errno,
                                  ngx_blocking_n " failed");
                    ngx_close_accepted_connection(c);
                    return;
                }
            }


        } else {
            if (!(ngx_event_flags & (NGX_USE_AIO_EVENT|NGX_USE_RTSIG_EVENT))) {
                if (ngx_nonblocking(s) == -1) {
                    ngx_log_error(NGX_LOG_ALERT, ev->log, ngx_socket_errno,
                                  ngx_nonblocking_n " failed");
                    ngx_close_accepted_connection(c);
                    return;
                }
            }
        }


比较特殊的是connect,unix平台的传统的做法都是让connect的socket进入非阻塞模式,NGINX也是这样,在ngx_event_connect_peer中,直接设置成非阻塞模式了事

    if (ngx_nonblocking(s) == -1) {
        ngx_log_error(NGX_LOG_ALERT, pc->log, ngx_socket_errno,
                      ngx_nonblocking_n " failed");


        goto failed;
    }


另外补充一点,win平台上的WsaAysncSelect以及WsaEventSelect会自动把套接口设置成非阻塞模式,因此不管connect,send,accept,recv都是非阻塞模式,并且有各自对应的通知消息类型


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值