日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Java實現權重隨機算法詳解

瀏覽:71日期:2022-08-09 08:34:02
目錄應用場景本文目標算法詳解權重比例Java 實現參考應用場景

客戶端負載均衡,例如 Nacos 提供的客戶端負載均衡就是使用了該算法游戲抽獎(普通道具的權重很高,稀有道具的權重很低)

本文目標

Java 實現權重隨機算法

算法詳解

比如我們現在有三臺 Server,權重分別為1,3,2。現在想對三臺 Server 做負載均衡

Server1 Server2 Server3 weight weight weight 1 3 2權重比例

我們算出每臺 Server 的權重比例,權重比例 = 自己的權重 / 總權重

server1 server2 server3 weight weight weight 1 3 2 radio radio radio 1/6 3/6 2/6

根據權重比例計算覆蓋區域

server1 server2 server3 ^ ^^ |---------||---------|---------|---------||---------|---------|| 0 1/6 4/6 6/6 ^ ^ ^ 0.16666667 0.66666667 1.0

根據權重負載均衡

如步驟2所示,每個 server 都有自己的范圍,把每一個格子作為單位來看的話

server1 (0,1] server2 (1,4] server3 (4,6]

使用隨機數函數,取 (0,6] 之間的隨機數,根據隨機數落在哪個范圍決定如何選擇。例如隨機數為 2,處于 (1,4] 范圍,那么就選擇 server2。

思路大概就是這樣,落實到代碼上,用一個數組 [0.16666667, 0.66666667, 1] 來表示這三個 server 的覆蓋范圍,使用 ThreadLocalRandom 或者 Random 獲取 [0,1) 內的隨機數。然后使用二分查找法快速定位隨機數處于哪個區間

Java 實現

代碼基本上與 com.alibaba.nacos.client.naming.utils.Chooser 一致,在可讀性方面做了下優化。

import java.util.*;import java.util.concurrent.ThreadLocalRandom;import java.util.concurrent.atomic.AtomicInteger;public class WeightRandom<T> { private final List<T> items = new ArrayList<>(); private double[] weights; public WeightRandom(List<ItemWithWeight<T>> itemsWithWeight) {this.calWeights(itemsWithWeight); } /** * 計算權重,初始化或者重新定義權重時使用 * */ public void calWeights(List<ItemWithWeight<T>> itemsWithWeight) {items.clear();// 計算權重總和double originWeightSum = 0;for (ItemWithWeight<T> itemWithWeight : itemsWithWeight) { double weight = itemWithWeight.getWeight(); if (weight <= 0) {continue; } items.add(itemWithWeight.getItem()); if (Double.isInfinite(weight)) {weight = 10000.0D; } if (Double.isNaN(weight)) {weight = 1.0D; } originWeightSum += weight;}// 計算每個item的實際權重比例double[] actualWeightRatios = new double[items.size()];int index = 0;for (ItemWithWeight<T> itemWithWeight : itemsWithWeight) { double weight = itemWithWeight.getWeight(); if (weight <= 0) {continue; } actualWeightRatios[index++] = weight / originWeightSum;}// 計算每個item的權重范圍// 權重范圍起始位置weights = new double[items.size()];double weightRangeStartPos = 0;for (int i = 0; i < index; i++) { weights[i] = weightRangeStartPos + actualWeightRatios[i]; weightRangeStartPos += actualWeightRatios[i];} } /** * 基于權重隨機算法選擇 * */ public T choose() {double random = ThreadLocalRandom.current().nextDouble();int index = Arrays.binarySearch(weights, random);if (index < 0) { index = -index - 1;} else { return items.get(index);}if (index < weights.length && random < weights[index]) { return items.get(index);}// 通常不會走到這里,為了保證能得到正確的返回,這里隨便返回一個return items.get(0); } public static class ItemWithWeight<T> {T item;double weight;public ItemWithWeight() {}public ItemWithWeight(T item, double weight) { this.item = item; this.weight = weight;}public T getItem() { return item;}public void setItem(T item) { this.item = item;}public double getWeight() { return weight;}public void setWeight(double weight) { this.weight = weight;} } public static void main(String[] args) {// for testint sampleCount = 1_000_000;ItemWithWeight<String> server1 = new ItemWithWeight<>('server1', 1.0);ItemWithWeight<String> server2 = new ItemWithWeight<>('server2', 3.0);ItemWithWeight<String> server3 = new ItemWithWeight<>('server3', 2.0);WeightRandom<String> weightRandom = new WeightRandom<>(Arrays.asList(server1, server2, server3));// 統計 (這里用 AtomicInteger 僅僅是因為寫起來比較方便,這是一個單線程測試)Map<String, AtomicInteger> statistics = new HashMap<>();for (int i = 0; i < sampleCount; i++) { statistics .computeIfAbsent(weightRandom.choose(), (k) -> new AtomicInteger()) .incrementAndGet();}statistics.forEach((k, v) -> { double hit = (double) v.get() / sampleCount; System.out.println(k + ', hit:' + hit);}); }}

這里重點說一下 Arrays.binarySearch(weights, random),這個 API 我之前沒有用過導致我在讀 Nacos 源碼時,對這塊的操作十分費解

來看一下 java API 文檔對該方法返回值的解釋

Returns:index of the search key, if it is contained in the array; otherwise, (-(insertion point) - 1). The insertion point is defined as the point at which the key would be inserted into the array: the index of the first element greater than the key, or a.length if all elements in the array are less than the specified key. Note that this guarantees that the return value will be >= 0 if and only if the key is found.

解釋下,首先該方法的作用是通過指定的 key 搜索數組。(前提條件是要保證數組的順序是從小到大排序過的)

如果數組中包含該 key,則返回對應的索引 如果不包含該 key,則返回該 key 的 (-(insertion point)-1)

insertion point(插入點):該 key 應該在數組的哪個位置。舉個例子,數組 [1,3,5],我的搜索 key 為 2,按照順序排的話 2 應該在數組的 index = 1 的位置,所以此時 insertion point = 1。

(這里 jdk 將能查到 key 和 查不到 key 兩種情況做了區分。為了將未找到的情況全部返回負數,所以做了 (-(insertion point)-1) 這樣的操作)

看到這,我們就懂了,insertion point 就是我們需要的,現在我們用小學數學來推導一下如何計算 insertion point

// 小學數學推導一下 insertion point 如何計算returnValue = (- (insertionPoint) - 1)insertionPoint = (- (returnValue + 1) )// 所以就有了上邊代碼中的if (index < 0) { index = -index - 1;}參考

https://github.com/alibaba/nacos/blob/develop/client/src/main/java/com/alibaba/nacos/client/naming/utils/Chooser.java

到此這篇關于Java實現權重隨機算法詳解的文章就介紹到這了,更多相關Java 權重隨機內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
精品一区毛片| 在线观看精品| 激情欧美丁香| 国产精品亚洲欧美| jizzjizz中国精品麻豆| 天堂√8在线中文| 一区二区电影在线观看| 精品视频在线你懂得| 亚洲专区视频| 欧美日韩视频免费观看| 久久精品xxxxx| 伊人影院久久| 亚州av一区| 国产精品一区二区中文字幕| 精品三级久久| 国产精选在线| 日韩精品一区二区三区中文| 久久av超碰| 国产精品美女| 亚洲精品综合| 最近高清中文在线字幕在线观看1| 91精品国产自产在线观看永久∴| 欧美亚洲综合视频| 91精品国产自产在线丝袜啪| 亚洲不卡系列| 国产精品中文字幕制服诱惑| 蜜芽一区二区三区| 日韩成人精品一区| 综合视频一区| 蜜臀国产一区| 久久久久久黄| 国产欧美一级| 视频在线观看一区二区三区| 国产日韩电影| 国产一区二区视频在线看| 日韩欧美精品一区| 国产亚洲一区二区三区啪| 黑人精品一区| 国产精品a久久久久| 亚洲精品福利| 日本大胆欧美人术艺术动态| 欧美精品导航| 国产精品中文字幕制服诱惑| 欧美精品中文| 亚洲一区二区三区高清不卡| 不卡视频在线| 视频二区不卡| 亚洲精品在线影院| 日韩精品亚洲一区二区三区免费| 不卡中文一二三区| 久久中文字幕导航| 国产精品对白久久久久粗| 日本а中文在线天堂| 国产极品嫩模在线观看91精品| 欧美精品一区二区三区精品| 最新日韩av| 亚洲在线网站| 美女网站久久| 亚洲综合五月| 亚洲一区二区三区在线免费| 水野朝阳av一区二区三区| 男人的天堂久久精品| 欧美成a人片免费观看久久五月天| 人人爱人人干婷婷丁香亚洲| 91亚洲精品视频在线观看| 尤物在线精品| 香蕉国产精品| 亚洲精品成人图区| 国产传媒在线观看| 日韩欧美午夜| 久久99高清| 麻豆久久一区| 国产在线观看91一区二区三区| 国产日韩欧美一区二区三区| 国产精品一区三区在线观看| 麻豆91精品视频| 欧美一级一区| 国产精品亚洲综合久久| 国产精品久久久久久久久久妞妞 | 欧美成a人国产精品高清乱码在线观看片在线观看久 | 亚洲二区三区不卡| 精品伊人久久| 国产精品任我爽爆在线播放| 国产精品香蕉| 麻豆免费精品视频| 国产无遮挡裸体免费久久| 日韩精品久久久久久| 91嫩草精品| 亚洲欧洲一区| 电影天堂国产精品| 美女精品在线| 精品久久久亚洲| 精品中文字幕一区二区三区av| 亚洲欧美网站| 另类综合日韩欧美亚洲| 91精品综合| 亚洲欧美在线综合| 婷婷综合成人| 国产一区二区三区视频在线| 在线日韩成人| 中文字幕av一区二区三区四区| 美女久久久精品| 欧美日中文字幕| 欧美一区二区三区高清视频| 亚洲人成在线影院| 精品国产一区二区三区av片| 黄色亚洲在线| 久久一二三区| 美女尤物国产一区| 91久久中文| 美女视频免费精品| 久久99伊人| 国产亚洲久久| 超碰在线99| 91精品综合| 久久99蜜桃| 久久成人亚洲| 免费国产自线拍一欧美视频| 国产一二在线播放| 日日夜夜免费精品| 久久久久久黄| 亚洲人成网站在线在线观看| 成人av三级| 欧美日韩精品一区二区三区视频| 国产精品一区高清| 丝袜美腿亚洲一区| 电影91久久久| 久久伊人久久| 最新国产精品久久久| 蜜桃精品在线| 亚洲h色精品| 欧美亚洲综合视频| 亚洲一区二区动漫| 美女免费视频一区| 亚洲欧美网站在线观看| 亚洲三级精品| 国产精品麻豆久久| 久久av一区| 久久精品国产亚洲一区二区三区| 国产99久久久国产精品成人免费| 少妇精品在线| 视频在线不卡免费观看| 日本在线啊啊| 三级久久三级久久久| 97人人精品| 亚洲三级毛片| 色婷婷狠狠五月综合天色拍| 蜜桃视频在线观看一区| 精品视频99| 免费看精品久久片| 91看片一区| 日韩成人午夜精品| 亚洲国产专区校园欧美| 啪啪亚洲精品| 99在线精品免费视频九九视 | 亚洲午夜免费| 国产精品久久久久久久久妇女| 不卡在线一区| 精品美女视频 | 红桃视频欧美| 精品国产一区二区三区噜噜噜| 日韩视频不卡| 成午夜精品一区二区三区软件| 日韩在线卡一卡二| 电影天堂国产精品| 国产精品99久久免费| 国产91精品对白在线播放| 国产福利一区二区精品秒拍| 亚洲欧美日本视频在线观看| 久久久久久色| 福利一区和二区| 日韩一区二区三区精品| 99精品美女| 精品女同一区二区三区在线观看| 男人的天堂亚洲一区| 久久一区二区三区喷水| 精品免费视频| 久久国产欧美日韩精品| 亚洲一区区二区| 久久久蜜桃一区二区人| 久久亚洲人体| 免费在线观看不卡| 激情91久久| 91精品xxx在线观看| 视频一区二区欧美| 日韩一区二区中文| 电影91久久久| 国产视频网站一区二区三区| 一本综合精品| 亚洲欧美日韩综合国产aⅴ| 亚洲高清av| 久久久五月天| 婷婷激情一区| 日本在线啊啊| 成年男女免费视频网站不卡| 精品美女久久| 91精品国产自产精品男人的天堂| 亚洲免费中文| 老司机久久99久久精品播放免费| 亚洲激情欧美|