【Nginx 源码学习】Nginx 架构设计与主流程分

}

}

ngx_log_debug0(NGX_LOG_DEBUG_EVENT, cycle->log, 0, “sigsuspend”);

/* 等待信号的到来,阻塞函数 */

sigsuspend(&set);

ngx_time_update();

ngx_log_debug1(NGX_LOG_DEBUG_EVENT, cycle->log, 0,

“wake up, sigio %i”, sigio);

/* 收到了SIGCHLD信号,有worker退出(ngx_reap == 1) */

if (ngx_reap) {

ngx_reap = 0;

ngx_log_debug0(NGX_LOG_DEBUG_EVENT, cycle->log, 0, “reap children”);

live = ngx_reap_children(cycle);

}

if (!live && (ngx_terminate || ngx_quit)) {

ngx_master_process_exit(cycle);

}

/* 中止进程 */

if (ngx_terminate) {

if (delay == 0) {

delay = 50;

}

if (sigio) {

sigio–;

continue;

}

sigio = ccf->worker_processes + 2 /* cache processes */;

if (delay > 1000) {

ngx_signal_worker_processes(cycle, SIGKILL);

} else {

ngx_signal_worker_processes(cycle,

ngx_signal_value(NGX_TERMINATE_SIGNAL));

}

continue;

}

/* 退出进程 */

if (ngx_quit) {

ngx_signal_worker_processes(cycle,

ngx_signal_value(NGX_SHUTDOWN_SIGNAL));

ls = cycle->listening.elts;

for (n = 0; n < cycle->listening.nelts; n++) {

if (ngx_close_socket(ls[n].fd) == -1) {

ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_socket_errno,

ngx_close_socket_n " %V failed", &ls[n].addr_text);

}

}

cycle->listening.nelts = 0;

continue;

}

/* 收到SIGHUP信号 重新初始化配置 */

if (ngx_reconfigure) {

ngx_reconfigure = 0;

if (ngx_new_binary) {

ngx_start_worker_processes(cycle, ccf->worker_processes,

NGX_PROCESS_RESPAWN);

ngx_start_cache_manager_processes(cycle, 0);

ngx_noaccepting = 0;

continue;

}

ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, “reconfiguring”);

cycle = ngx_init_cycle(cycle);

if (cycle == NULL) {

cycle = (ngx_cycle_t *) ngx_cycle;

continue;

}

ngx_cycle = cycle;

ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx,

ngx_core_module);

ngx_start_worker_processes(cycle, ccf->worker_processes,

NGX_PROCESS_JUST_RESPAWN);

ngx_start_cache_manager_processes(cycle, 1);

/* allow new processes to start */

ngx_msleep(100);

live = 1;

ngx_signal_worker_processes(cycle,

ngx_signal_value(NGX_SHUTDOWN_SIGNAL));

}

/* 当ngx_noaccepting==1时,会把ngx_restart设为1,重启worker */

if (ngx_restart) {

ngx_restart = 0;

ngx_start_worker_processes(cycle, ccf->worker_processes,

NGX_PROCESS_RESPAWN);

ngx_start_cache_manager_processes(cycle, 0);

live = 1;

}

/* 收到SIGUSR1信号,重新打开log文件 */

if (ngx_reopen) {

ngx_reopen = 0;

ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, “reopening logs”);

ngx_reopen_files(cycle, ccf->user);

ngx_signal_worker_processes(cycle,

ngx_signal_value(NGX_REOPEN_SIGNAL));

}

/* SIGUSER2,热代码替换 */

if (ngx_change_binary) {

ngx_change_binary = 0;

ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, “changing binary”);

ngx_new_binary = ngx_exec_new_binary(cycle, ngx_argv);

}

/* 收到SIGWINCH信号不在接受请求,worker退出,master不退出 */

if (ngx_noaccept) {

ngx_noaccept = 0;

ngx_noaccepting = 1;

ngx_signal_worker_processes(cycle,

ngx_signal_value(NGX_SHUTDOWN_SIGNAL));

}

}

}

抽出主要流程看的比较清楚:

void ngx_master_process_cycle(ngx_cycle_t *cycle) {

···

// 启动各个worker进程

ngx_start_worker_processes(cycle, ccf->worker_processes, NGX_PROCESS_RESPAWN);

···

for (;😉 {

···

if (···) {

// 这里主要是向各个进程发送命令

ngx_signal_worker_processes(cycle, ngx_signal_value(NGX_SHUTDOWN_SIGNAL));

}

}

}


ngx_start_worker_processes 创建工作进程

先了解一下 ngx_channel_t,有点重要哈:

master进程每次发送给worker进程的指令用如下的一个结构来完成封装:

typedef struct {

// 传递的 TCP 消息中的命令

ngx_uint_t command;

// 进程 ID,一般是发送命令方的进程 ID

ngx_pid_t pid;

// 表示发送命令方在 ngx_processes 进程数组间的序号

ngx_int_t slot;

// 通信的套接字句柄

ngx_fd_t fd;

}ngx_channel_t;

Nginx 针对 command 成员定义了如下命令:

// 打开频道,使用频道这种方式通信前必须发送的命令

#define NGX_CMD_OPEN_CHANNEL 1

// 关闭已经打开的频道,实际上也就是关闭套接字

#define NGX_CMD_CLOSE_CHANNEL 2

// 要求接收方正常地退出进程

#define NGX_CMD_QUIT 3

// 要求接收方强制地结束进程

#define NGX_CMD_TERMINATE 4

// 要求接收方重新打开进程已经打开过的文件

#define NGX_CMD_REOPEN 5

/**

  • 创建工作进程

*/

static void ngx_start_worker_processes(ngx_cycle_t *cycle, ngx_int_t n,

ngx_int_t type) {

ngx_int_t i;

ngx_channel_t ch;

ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, “start worker processes”);

ngx_memzero(&ch, sizeof(ngx_channel_t));

ch.command = NGX_CMD_OPEN_CHANNEL;

/* 循环创建工作进程 默认ccf->worker_processes=8个进程,根据CPU个数决定 */

for (i = 0; i < n; i++) {

/* 打开工作进程 (ngx_worker_process_cycle 回调函数,主要用于处理每个工作线程)*/

ngx_spawn_process(cycle, ngx_worker_process_cycle,

(void *) (intptr_t) i, “worker process”, type);

ch.pid = ngx_processes[ngx_process_slot].pid;

ch.slot = ngx_process_slot;

ch.fd = ngx_processes[ngx_process_slot].channel[0];

ngx_pass_open_channel(cycle, &ch);

}

}

简写一下:

static void ngx_start_worker_processes(ngx_cycle_t *cycle, ngx_int_t n, ngx_int_t type) {

···

ngx_channel_t ch;

···

ch.command = NGX_CMD_OPEN_CHANNEL; //当前是新建了一个进程

for (i = 0; i < n; i++) {

//spawn,生成一个子进程

// ngx_worker_process_cycle:该子进程所进行的事件循环,这里先不管它什么循环

// worker进程在一个无限for循环中,不断的检查相应的事件模型中是否存在对应的事件,

// 然后将accept事件和read、write事件分开放入两个队列中,最后在事件循环中不断的处理事件

ngx_spawn_process(cycle, ngx_worker_process_cycle,

(void *) (intptr_t) i, “worker process”, type);

// 下面的这段代码的主要作用是将新建进程这个事件通知到其他的进程,

// 其就会向ngx_processes数组的每个进程的channel[0]上写入当前广播的事件,也即这里的ch,

// 因为子进程之间也需要通信

ch.pid = ngx_processes[ngx_process_slot].pid;

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Java)
img

文末

我将这三次阿里面试的题目全部分专题整理出来,并附带上详细的答案解析,生成了一份PDF文档

  • 第一个要分享给大家的就是算法和数据结构

网易严选Java开发三面面经:HashMap+JVM+索引+消息队列

  • 第二个就是数据库的高频知识点与性能优化

网易严选Java开发三面面经:HashMap+JVM+索引+消息队列

  • 第三个则是并发编程(72个知识点学习)

网易严选Java开发三面面经:HashMap+JVM+索引+消息队列

  • 最后一个是各大JAVA架构专题的面试点+解析+我的一些学习的书籍资料

网易严选Java开发三面面经:HashMap+JVM+索引+消息队列

还有更多的Redis、MySQL、JVM、Kafka、微服务、Spring全家桶等学习笔记这里就不一一列举出来

一个人可以走的很快,但一群人才能走的更远。如果你从事以下工作或对以下感兴趣,欢迎戳这里加入程序员的圈子,让我们一起学习成长!

AI人工智能、Android移动开发、AIGC大模型、C C#、Go语言、Java、Linux运维、云计算、MySQL、PMP、网络安全、Python爬虫、UE5、UI设计、Unity3D、Web前端开发、产品经理、车载开发、大数据、鸿蒙、计算机网络、嵌入式物联网、软件测试、数据结构与算法、音视频开发、Flutter、IOS开发、PHP开发、.NET、安卓逆向、云计算

  • 第三个则是并发编程(72个知识点学习)

[外链图片转存中…(img-k90a4aLf-1712274326318)]

  • 最后一个是各大JAVA架构专题的面试点+解析+我的一些学习的书籍资料

[外链图片转存中…(img-7yN1tkGW-1712274326318)]

还有更多的Redis、MySQL、JVM、Kafka、微服务、Spring全家桶等学习笔记这里就不一一列举出来

一个人可以走的很快,但一群人才能走的更远。如果你从事以下工作或对以下感兴趣,欢迎戳这里加入程序员的圈子,让我们一起学习成长!

AI人工智能、Android移动开发、AIGC大模型、C C#、Go语言、Java、Linux运维、云计算、MySQL、PMP、网络安全、Python爬虫、UE5、UI设计、Unity3D、Web前端开发、产品经理、车载开发、大数据、鸿蒙、计算机网络、嵌入式物联网、软件测试、数据结构与算法、音视频开发、Flutter、IOS开发、PHP开发、.NET、安卓逆向、云计算

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值