From 85d110b55ab55c41aba1ed690062ebf29412a6f9 Mon Sep 17 00:00:00 2001 From: teaey Date: Fri, 18 Mar 2016 22:07:31 +0800 Subject: [PATCH] fix #215 --- .../loadbalance/RoundRobinLoadBalance.java | 133 ++++++++++-------- 1 file changed, 77 insertions(+), 56 deletions(-) diff --git a/dubbo-cluster/src/main/java/com/alibaba/dubbo/rpc/cluster/loadbalance/RoundRobinLoadBalance.java b/dubbo-cluster/src/main/java/com/alibaba/dubbo/rpc/cluster/loadbalance/RoundRobinLoadBalance.java index ff65499be76..19443b8dd3f 100644 --- a/dubbo-cluster/src/main/java/com/alibaba/dubbo/rpc/cluster/loadbalance/RoundRobinLoadBalance.java +++ b/dubbo-cluster/src/main/java/com/alibaba/dubbo/rpc/cluster/loadbalance/RoundRobinLoadBalance.java @@ -13,10 +13,11 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package com.alibaba.dubbo.rpc.cluster.loadbalance; - -import java.util.ArrayList; +package com.alibaba.dubbo.rpc.cluster.loadbalance; + +import java.util.LinkedHashMap; import java.util.List; +import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentMap; @@ -24,59 +25,79 @@ import com.alibaba.dubbo.common.utils.AtomicPositiveInteger; import com.alibaba.dubbo.rpc.Invocation; import com.alibaba.dubbo.rpc.Invoker; - -/** - * Round robin load balance. - * - * @author qian.lei - * @author william.liangf - */ + +/** + * Round robin load balance. + * + * @author qian.lei + * @author william.liangf + */ public class RoundRobinLoadBalance extends AbstractLoadBalance { - - public static final String NAME = "roundrobin"; - - private final ConcurrentMap sequences = new ConcurrentHashMap(); - private final ConcurrentMap weightSequences = new ConcurrentHashMap(); - - protected Invoker doSelect(List> invokers, URL url, Invocation invocation) { - String key = invokers.get(0).getUrl().getServiceKey() + "." + invocation.getMethodName(); - int length = invokers.size(); // 总个数 - int maxWeight = 0; // 最大权重 - int minWeight = Integer.MAX_VALUE; // 最小权重 - for (int i = 0; i < length; i++) { - int weight = getWeight(invokers.get(i), invocation); - maxWeight = Math.max(maxWeight, weight); // 累计最大权重 - minWeight = Math.min(minWeight, weight); // 累计最小权重 - } - if (maxWeight > 0 && minWeight < maxWeight) { // 权重不一样 - AtomicPositiveInteger weightSequence = weightSequences.get(key); - if (weightSequence == null) { - weightSequences.putIfAbsent(key, new AtomicPositiveInteger()); - weightSequence = weightSequences.get(key); - } - int currentWeight = weightSequence.getAndIncrement() % maxWeight; - List> weightInvokers = new ArrayList>(); - for (Invoker invoker : invokers) { // 筛选权重大于当前权重基数的Invoker - if (getWeight(invoker, invocation) > currentWeight) { - weightInvokers.add(invoker); - } - } - int weightLength = weightInvokers.size(); - if (weightLength == 1) { - return weightInvokers.get(0); - } else if (weightLength > 1) { - invokers = weightInvokers; - length = invokers.size(); - } - } - AtomicPositiveInteger sequence = sequences.get(key); - if (sequence == null) { - sequences.putIfAbsent(key, new AtomicPositiveInteger()); - sequence = sequences.get(key); - } - // 取模轮循 - return invokers.get(sequence.getAndIncrement() % length); - } - + public static final String NAME = "roundrobin"; + + private final ConcurrentMap sequences = new ConcurrentHashMap(); + + private static final class IntegerWrapper { + public IntegerWrapper(int value) { + this.value = value; + } + + private int value; + + public int getValue() { + return value; + } + + public void setValue(int value) { + this.value = value; + } + + public void decrement() { + this.value--; + } + } + + protected Invoker doSelect(List> invokers, URL url, Invocation invocation) { + String key = invokers.get(0).getUrl().getServiceKey() + "." + invocation.getMethodName(); + int length = invokers.size(); // 总个数 + int maxWeight = 0; // 最大权重 + int minWeight = Integer.MAX_VALUE; // 最小权重 + final LinkedHashMap, IntegerWrapper> invokerToWeightMap = new LinkedHashMap, IntegerWrapper>(); + int weightSum = 0; + for (int i = 0; i < length; i++) { + int weight = getWeight(invokers.get(i), invocation); + maxWeight = Math.max(maxWeight, weight); // 累计最大权重 + minWeight = Math.min(minWeight, weight); // 累计最小权重 + if (weight > 0) { + invokerToWeightMap.put(invokers.get(i), new IntegerWrapper(weight)); + weightSum += weight; + } + } + AtomicPositiveInteger sequence = sequences.get(key); + if (sequence == null) { + sequences.putIfAbsent(key, new AtomicPositiveInteger()); + sequence = sequences.get(key); + } + int currentSequence = sequence.getAndIncrement(); + if (maxWeight > 0 && minWeight < maxWeight) { // 权重不一样 + int mod = currentSequence % weightSum; + for (int i = 0; i < maxWeight; i++) { + for (Map.Entry, IntegerWrapper> each : invokerToWeightMap.entrySet()) { + final Invoker k = each.getKey(); + final IntegerWrapper v = each.getValue(); + if (mod == 0 && v.getValue() > 0) { + return k; + } + if (v.getValue() > 0) { + v.decrement(); + mod--; + } + } + } + } + // 取模轮循 + return invokers.get(currentSequence % length); + } + } \ No newline at end of file