首頁 運維 Nginx Nginx如何實作輪詢演算法

Nginx如何實作輪詢演算法

May 21, 2023 pm 09:43 PM
nginx

簡單輪詢演算法

這種演算法比較簡單,舉例就是你有三台伺服器

第一台伺服器 192.168.1.1
#第二台伺服器 192.168.1.2
#第三台伺服器 192.168.1.3

#第一個請求過來之後預設存取第一台,第二個請求過來存取第二台,第三次請求過來訪問第三台,第四次請求過來訪問第一台,以此類推。以下是我程式碼實作簡單得演算法:

public class simplepolling {

  /**
   * key是ip
   */
  public static list <string> ipservice = new linkedlist <>();
  static {
    ipservice.add("192.168.1.1");
    ipservice.add("192.168.1.2");
    ipservice.add("192.168.1.3");
  }
  public static int pos = 0;
  public static string getip(){
    if(pos >= ipservice.size()){
      //防止索引越界
      pos = 0;
    }
    string ip = ipservice.get(pos);
    pos ++;
    return ip;

  }

  public static void main(string[] args) {
    for (int i = 0; i < 4; i++) {
      system.out.println(getip());

    }
  }
}
登入後複製

模擬執行4次執行結果是

Nginx如何實作輪詢演算法

此時如果我有一台伺服器效能比較好(例如192.168.1.1),我想讓這台伺服器處理多一點請求,此時就涉及到了權重得概率,這種演算法就不能實現,請看我後面描述的輪詢升級版演算法。

加權輪詢演算法

此時我需要把我前面3台伺服器都設定權重,例如第一台設定5,第二台設定1,第三台設定1

第一台伺服器192.168.1.15
#第二台伺服器192.168.1.21
第三台伺服器192.168.1.31

此時前5個請求都會存取到第一台伺服器,第六個請求會存取到第二台伺服器,第七個請求會訪問到第三台伺服器。

以下是我給的程式碼案例:

public class weightpolling {

  /**
   * key是ip,value是权重
   */
  public static map<string, integer> ipservice = new linkedhashmap<>();
  static {
    ipservice.put("192.168.1.1", 5);
    ipservice.put("192.168.1.2", 1);
    ipservice.put("192.168.1.3", 1);
  }
  public static int requestid = 0;
  public static int getandincrement() {
    return requestid++;
  }

  public static string getip(){
    //获取总的权重
    int totalweight =0;
    for (integer value : ipservice.values()) {
      totalweight+= value;
    }
    //获取当前轮询的值
    int andincrement = getandincrement();
    int pos = andincrement% totalweight;
    for (string ip : ipservice.keyset()) {
      if(pos < ipservice.get(ip)){
        return ip;
      }
      pos -= ipservice.get(ip);
    }
    return null;
  }

  public static void main(string[] args) {
    for (int i = 0; i < 7; i++) {
      system.out.println(getip());
    }
  }

}
登入後複製

此時執行結果是

Nginx如何實作輪詢演算法

可以看的第一台伺服器執行了5次,後面2台依序執行一次,依序類推。可能你覺得這種演算法還不錯。其實這種演算法有一個缺點是,如果我第一台伺服器設定權重過大可能我需要很多次請求都執行到第一台伺服器上去,這樣的情況分佈是不均勻的,會造成某一台伺服器壓力過大導致崩潰。所以我後面要引入第三種演算法來解決這個問題

平滑加權輪詢演算法

這個演算法可能比較複雜,我第一次看也有點不太明白,後面看過相關資料在結合我自己的理解給大家圖文解釋一下,這裡我舉例的伺服器配置和權重還是跟上面一樣

##6# 192.168.1.1{-1,-3,4}#5##6#{9,-1,-1}79#192.168.1.1{2,-1,-1}7{7,0,0}77192.168.1.1{0,0,0}由上圖可以看出第一台伺服器雖然權重設定的是5,但不是第五次請求過來都是第一台伺服器執行,而是分散執行,調度序列是非常均勻的,且第7 次調度時選中後當前權重又回到{0, 0, 0},實例的狀態同初始狀態一致,所以後續可以一直重複調度操作。
請求目前權重= 自身權重選中後目前權重總權重#目前最大權重傳回的ip選取後當前權重=目前最大權重-總權重
1{5,1,1}#7 5192.168.1.1{-2,1,1}
2{3,2 ,2}73192.168.1.1{-4,2,2}
3{1,3,3}73192.168.1.2{1,-4 ,3}
4{6,-3,4}7
##{4,-2,5} #75192.168.1.3#{4,-2,-2}
可能有的人還不能清楚的明白上一張圖表示的含義,我這裡大概描述一下:

1.首先總權重不會變,預設就是當前設定的權重之和

2.在第一次請求進來的時候我預設初始化目前權重選值是{0,0,0},所以當前權重的值就是{5 0,1 0,1 0} ,這裡的5,1,1就是我們前面每台伺服器設定的權重。

3.這裡我們可以得到第一次請求過來的最大權重是5。然後返回第一台伺服器ip

4.然後我們設定選中後當前權重,這裡就是當前最大權重減去總權重(5-7),沒有選中的權重不變,這時候得到當前權重選取權重的值{5-7,1,1}

5.在第二次請求過來的時候我們延續上面的2,3,4步驟執行.

如果這裡還有不懂我下面會提供我自己用java程式碼實作的演算法:

public class polling {

  /**
   * key是ip,value是权重
   */
  public static map <string,integer> ipservice = new linkedhashmap <>();
  static {
    ipservice.put("192.168.1.1",5);
    ipservice.put("192.168.1.2",1);
    ipservice.put("192.168.1.3",1);
  }
  private static map<string,weight> weightmap = new linkedhashmap <>();

  public static string getip(){
    //计算总的权重
     int totalweight = 0;
    for (integer value : ipservice.values()) {
      totalweight+=value;
    }
    //首先判断weightmap是否为空
    if(weightmap.isempty()){
      ipservice.foreach((ip,weight)->{
        weight weights = new weight(ip, weight,0);
        weightmap.put(ip,weights);
      });
    }
    //给map中得对象设置当前权重
    weightmap.foreach((ip,weight)->{
      weight.setcurrentweight(weight.getweight() + weight.getcurrentweight());
    });

    //判断最大权重是否大于当前权重,如果为空或者小于当前权重,则把当前权重赋值给最大权重
    weight maxweight = null;
    for (weight weight : weightmap.values()) {
      if(maxweight ==null || weight.getcurrentweight() > maxweight.getcurrentweight()){
        maxweight = weight;
      }
    }
    //最后把当前最大权重减去总的权重
    maxweight.setcurrentweight(maxweight.getcurrentweight() - totalweight);
    //返回
    return maxweight.getip();
  }

  public static void main(string[] args) {
    //模拟轮询7次取ip
    for (int i = 0; i < 7; i++) {
      system.out.println(getip());
    }
  }

}

class weight{
  /**
   * ip
   */
  private string ip;
  /**
   * 设置得权重
   */
  private int weight;
  /**
   * 当前权重
   */
  private int currentweight;

  public weight(string ip, int weight,int currentweight) {
    this.ip = ip;
    this.weight = weight;
    this.currentweight = currentweight;
  }

  public string getip() {
    return ip;
  }

  public void setip(string ip) {
    this.ip = ip;
  }

  public int getweight() {
    return weight;
  }

  public void setweight(int weight) {
    this.weight = weight;
  }

  public int getcurrentweight() {
    return currentweight;
  }

  public void setcurrentweight(int currentweight) {
    this.currentweight = currentweight;
  }
}
登入後複製
這裡程式碼得執行結果是:


可以看出此處執行結果和表格裡描述得結果一致。 Nginx如何實作輪詢演算法

以上是Nginx如何實作輪詢演算法的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

nginx在windows中怎麼配置 nginx在windows中怎麼配置 Apr 14, 2025 pm 12:57 PM

如何在 Windows 中配置 Nginx?安裝 Nginx 並創建虛擬主機配置。修改主配置文件並包含虛擬主機配置。啟動或重新加載 Nginx。測試配置並查看網站。選擇性啟用 SSL 並配置 SSL 證書。選擇性設置防火牆允許 80 和 443 端口流量。

docker容器名稱怎麼查 docker容器名稱怎麼查 Apr 15, 2025 pm 12:21 PM

可以通過以下步驟查詢 Docker 容器名稱:列出所有容器(docker ps)。篩選容器列表(使用 grep 命令)。獲取容器名稱(位於 "NAMES" 列中)。

docker怎麼啟動容器 docker怎麼啟動容器 Apr 15, 2025 pm 12:27 PM

Docker 容器啟動步驟:拉取容器鏡像:運行 "docker pull [鏡像名稱]"。創建容器:使用 "docker create [選項] [鏡像名稱] [命令和參數]"。啟動容器:執行 "docker start [容器名稱或 ID]"。檢查容器狀態:通過 "docker ps" 驗證容器是否正在運行。

怎麼查看nginx是否啟動 怎麼查看nginx是否啟動 Apr 14, 2025 pm 01:03 PM

確認 Nginx 是否啟動的方法:1. 使用命令行:systemctl status nginx(Linux/Unix)、netstat -ano | findstr 80(Windows);2. 檢查端口 80 是否開放;3. 查看系統日誌中 Nginx 啟動消息;4. 使用第三方工具,如 Nagios、Zabbix、Icinga。

nginx怎麼查版本 nginx怎麼查版本 Apr 14, 2025 am 11:57 AM

可以查詢 Nginx 版本的方法有:使用 nginx -v 命令;查看 nginx.conf 文件中的 version 指令;打開 Nginx 錯誤頁,查看頁面的標題。

docker怎麼創建容器 docker怎麼創建容器 Apr 15, 2025 pm 12:18 PM

在 Docker 中創建容器: 1. 拉取鏡像: docker pull [鏡像名] 2. 創建容器: docker run [選項] [鏡像名] [命令] 3. 啟動容器: docker start [容器名]

nginx怎麼配置雲服務器域名 nginx怎麼配置雲服務器域名 Apr 14, 2025 pm 12:18 PM

在雲服務器上配置 Nginx 域名的方法:創建 A 記錄,指向雲服務器的公共 IP 地址。在 Nginx 配置文件中添加虛擬主機塊,指定偵聽端口、域名和網站根目錄。重啟 Nginx 以應用更改。訪問域名測試配置。其他注意事項:安裝 SSL 證書啟用 HTTPS、確保防火牆允許 80 端口流量、等待 DNS 解析生效。

怎麼啟動nginx服務器 怎麼啟動nginx服務器 Apr 14, 2025 pm 12:27 PM

啟動 Nginx 服務器需要按照不同操作系統採取不同的步驟:Linux/Unix 系統:安裝 Nginx 軟件包(例如使用 apt-get 或 yum)。使用 systemctl 啟動 Nginx 服務(例如 sudo systemctl start nginx)。 Windows 系統:下載並安裝 Windows 二進製文件。使用 nginx.exe 可執行文件啟動 Nginx(例如 nginx.exe -c conf\nginx.conf)。無論使用哪種操作系統,您都可以通過訪問服務器 IP

See all articles