深入理解nginx负载均衡round-robin算法

2024-03-07 12:04

本文主要是介绍深入理解nginx负载均衡round-robin算法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 1. 概述
  • 2. 如何启用round-robin算法
  • 3. 初始化round-robin算法
    • 3.1 设置算法上下文环境初始化回调函数
    • 3.2 加载服务器列表
  • 4. 初始化负载均衡请求上下文
  • 5. 获取peer
  • 6. 释放peer

1. 概述

   nginx为我们提供了强大的HTTP代理功能,而负载均衡算法是实现后端多Real Server代理的一个重要环节,nginx内部为我们集成了一些负载均衡算法,包括round-robin、ip-hash、hash、least_con、random等算法,另外第三方模块甚至提供了一致性哈希等算法。其中最基础的算法要数round-robin了,它也是nginx默认采用的负载均衡算法。本文透过nginx的源码来深入理解一下其实现机制,并对nginx在启动代理逻辑的时候如何进行Real server的选择过程有一个深入的理解。

   在开始之前,先来了解一下什么是round-robin算法,round-robin算法中文名叫轮询调度算法,就是以轮询的方式依次将请求调度不同的服务器,即每次调度执行i = (i + 1) mod n,并选出第i台服务器。算法的优点是其简洁性,它无需记录当前所有连接的状态,所以它是一种无状态调度。轮询调度算法假设所有服务器的处理性能都相同,不关心每台服务器的当前连接数和响应速度。当每台服务器的能力有差异的情况下,轮询调度算法容易导致实际服务器间的负载不平衡。

  为了克服round-robin算法的缺点,所以又在此基础上,衍生出了weighted round-robin算法,即加权轮询调度算法,它可以为不同的Real Server设置不同的权重,权重高的Real Server会分配到更多的负载。而nginx实现的round-robin算法实际上是weighted round-robin算法。

  以下就从源码层面来对nginx的round-robin算法进行分析。

2. 如何启用round-robin算法

  因为round-robin是nginx的默认负载均衡算法,因此如果在nginx配置文件的upstream块中如果没有设置任何其他的负载均衡算法,那么就自然启用round-robin算法了。

  因为nginx实际实现的是weighted round-robin算法,因此,可以对每个服务器设置其负载权重,如下:

upstream {server 192.168.0.1 weight=1;server 192.168.0.2 weight=2;server 192.168.0.3 weight=4;
......
};

  在upstream的初始化代码ngx_http_upstream_init_main_conf函数(这个函数是在配置读取完毕,并且在配置merge之前)中执行的,代码如下:

static char *
ngx_http_upstream_init_main_conf(ngx_conf_t *cf, void *conf)
{ngx_http_upstream_main_conf_t  *umcf = conf;ngx_uint_t                      i;ngx_array_t                     headers_in;ngx_hash_key_t                 *hk;ngx_hash_init_t                 hash;ngx_http_upstream_init_pt       init;ngx_http_upstream_header_t     *header;ngx_http_upstream_srv_conf_t  **uscfp;uscfp = umcf->upstreams.elts;for (i = 0; i < umcf->upstreams.nelts; i++) {init = uscfp[i]->peer.init_upstream ? uscfp[i]->peer.init_upstream:ngx_http_upstream_init_round_robin;if (init(cf, uscfp[i]) != NGX_OK) {return NGX_CONF_ERROR;}}
......
}

  可以看到,如果配置解析完毕后,upstream块对应的负载均衡算法没有被设置成任何其他负载均衡算法,那么就设置成默认的round-rpbin算法,下面就是调用ngx_http_upstream_init_round_robin函数进行算法的初始化了。

3. 初始化round-robin算法

3.1 设置算法上下文环境初始化回调函数

  首先,在ngx_http_upstream_init_round_robin函数中,首先执行了下面这个语句:

us->peer.init = ngx_http_upstream_init_round_robin_peer;

   这个ngx_http_upstream_init_round_robin_peer回调函数用来在nginx收到请求后,在进行执行负载均衡算法前,对round-robin算法的上下文环境进行初始化。

3.2 加载服务器列表

  在nginx配置文件解析到upstream的时候,server列表已经被加载到ngx_http_upstream_srv_conf_t的servers数组中了,在ngx_http_upstream_init_round_robin函数中,需要将加载到的每一个server放入ngx_http_upstream_rr_peer_t中,并进行一些初始化工作。nginx将配置的server分为两组,一个是主server,一个是backup server,其中默认使用的是主server列表中的服务器,只有到主server中的服务器都不可用了才会考虑使用backup server。 nginx将主server和backup server分两次来进行加载,首先加载主server。

  第一步险要计算server的总全重w、总的地址数n和可用地址数t,源码如下:

   for (i = 0; i < us->servers->nelts; i++) {/* 这里仅加载主server,所以backup server直接跳过 */if (server[i].backup) {    continue;}n += server[i].naddrs;w += server[i].naddrs * server[i].weight;if (!server[i].down) {t += server[i].naddrs;}}

  第二步是创建ngx_http_upstream_rr_peer_t结构体并进行初始化

  ngx_http_upstream_rr_peer_t结构体保存了每个peer的信息,和相关描述信息,如总的权重、总的地址数、域名等。源码如下:

	peers = ngx_pcalloc(cf->pool, sizeof(ngx_http_upstream_rr_peers_t));if (peers == NULL) {return NGX_ERROR;}......peer = ngx_pcalloc(cf->pool, sizeof(ngx_http_upstream_rr_peer_t) * n);if (peer == NULL) {return NGX_ERROR;}peers->single = (n == 1);       /* single=1表示仅有一个rs地址 */peers->number = n;peers->weighted = (w != n);peers->total_weight = w

这篇关于深入理解nginx负载均衡round-robin算法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/783452

相关文章

MySQL数据库约束深入详解

《MySQL数据库约束深入详解》:本文主要介绍MySQL数据库约束,在MySQL数据库中,约束是用来限制进入表中的数据类型的一种技术,通过使用约束,可以确保数据的准确性、完整性和可靠性,需要的朋友... 目录一、数据库约束的概念二、约束类型三、NOT NULL 非空约束四、DEFAULT 默认值约束五、UN

Java Stream流使用案例深入详解

《JavaStream流使用案例深入详解》:本文主要介绍JavaStream流使用案例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录前言1. Lambda1.1 语法1.2 没参数只有一条语句或者多条语句1.3 一个参数只有一条语句或者多

Nginx location匹配模式与规则详解

《Nginxlocation匹配模式与规则详解》:本文主要介绍Nginxlocation匹配模式与规则,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、环境二、匹配模式1. 精准模式2. 前缀模式(不继续匹配正则)3. 前缀模式(继续匹配正则)4. 正则模式(大

springboot上传zip包并解压至服务器nginx目录方式

《springboot上传zip包并解压至服务器nginx目录方式》:本文主要介绍springboot上传zip包并解压至服务器nginx目录方式,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录springboot上传zip包并解压至服务器nginx目录1.首先需要引入zip相关jar包2.然

如何使用Nginx配置将80端口重定向到443端口

《如何使用Nginx配置将80端口重定向到443端口》这篇文章主要为大家详细介绍了如何将Nginx配置为将HTTP(80端口)请求重定向到HTTPS(443端口),文中的示例代码讲解详细,有需要的小伙... 目录1. 创建或编辑Nginx配置文件2. 配置HTTP重定向到HTTPS3. 配置HTTPS服务器

深入理解Apache Kafka(分布式流处理平台)

《深入理解ApacheKafka(分布式流处理平台)》ApacheKafka作为现代分布式系统中的核心中间件,为构建高吞吐量、低延迟的数据管道提供了强大支持,本文将深入探讨Kafka的核心概念、架构... 目录引言一、Apache Kafka概述1.1 什么是Kafka?1.2 Kafka的核心概念二、Ka

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各

Nginx中配置HTTP/2协议的详细指南

《Nginx中配置HTTP/2协议的详细指南》HTTP/2是HTTP协议的下一代版本,旨在提高性能、减少延迟并优化现代网络环境中的通信效率,本文将为大家介绍Nginx配置HTTP/2协议想详细步骤,需... 目录一、HTTP/2 协议概述1.HTTP/22. HTTP/2 的核心特性3. HTTP/2 的优

springboot+dubbo实现时间轮算法

《springboot+dubbo实现时间轮算法》时间轮是一种高效利用线程资源进行批量化调度的算法,本文主要介绍了springboot+dubbo实现时间轮算法,文中通过示例代码介绍的非常详细,对大家... 目录前言一、参数说明二、具体实现1、HashedwheelTimer2、createWheel3、n

详解nginx 中location和 proxy_pass的匹配规则

《详解nginx中location和proxy_pass的匹配规则》location是Nginx中用来匹配客户端请求URI的指令,决定如何处理特定路径的请求,它定义了请求的路由规则,后续的配置(如... 目录location 的作用语法示例:location /www.chinasem.cntestproxy