摘要:失败安全,出现异常时,直接忽略。失败自动恢复,在调用失败后,返回一个空结果给服务提供者。源码分析一该类实现了接口,是集群的抽象类。
集群——cluster
目标:介绍dubbo中集群容错的几种模式,介绍dubbo-cluster下support包的源码。前言
集群容错还是很好理解的,就是当你调用失败的时候所作出的措施。先来看看有哪些模式:
图有点小,见谅,不过可以眯着眼睛看稍微能看出来一点,每一个Cluster实现类都对应着一个invoker,因为这个模式启用的时间点就是在调用的时候,而我在之前的文章里面讲过,invoker贯穿来整个服务的调用。不过这里除了调用失败的一些模式外,还有几个特别的模式,他们应该说成是失败的措施,而已调用的方式。
Failsafe Cluster:失败安全,出现异常时,直接忽略。失败安全就是当调用过程中出现异常时,FailsafeClusterInvoker 仅会打印异常,而不会抛出异常。适用于写入审计日志等操作
Failover Cluster:失败自动切换,当调用出现失败的时候,会自动切换集群中其他服务器,来获得invoker重试,通常用于读操作,但重试会带来更长延迟。一般都会设置重试次数。
Failfast Cluster:只会进行一次调用,失败后立即抛出异常。适用于幂等操作,比如新增记录。
Failback Cluster:失败自动恢复,在调用失败后,返回一个空结果给服务提供者。并通过定时任务对失败的调用记录并且重传,适合执行消息通知等操作。
Forking Cluster:会在线程池中运行多个线程,来调用多个服务器,只要一个成功即返回。通常用于实时性要求较高的读操作,但需要浪费更多服务资源。一般会设置最大并行数。
Available Cluster:调用第一个可用的服务器,仅仅应用于多注册中心。
Broadcast Cluster:广播调用所有提供者,逐个调用,在循环调用结束后,只要任意一台报错就报错。通常用于通知所有提供者更新缓存或日志等本地资源信息
Mergeable Cluster:该部分在分组聚合讲述。
MockClusterWrapper:该部分在本地伪装讲述。
源码分析 (一)AbstractClusterInvoker该类实现了Invoker接口,是集群Invoker的抽象类。
1.属性private static final Logger logger = LoggerFactory .getLogger(AbstractClusterInvoker.class); /** * 目录,包含多个invoker */ protected final Directory2.selectdirectory; /** * 是否需要核对可用 */ protected final boolean availablecheck; /** * 是否销毁 */ private AtomicBoolean destroyed = new AtomicBoolean(false); /** * 粘滞连接的Invoker */ private volatile Invoker stickyInvoker = null;
protected Invokerselect(LoadBalance loadbalance, Invocation invocation, List > invokers, List > selected) throws RpcException { // 如果invokers为空,则返回null if (invokers == null || invokers.isEmpty()) return null; // 获得方法名 String methodName = invocation == null ? "" : invocation.getMethodName(); // 是否启动了粘滞连接 boolean sticky = invokers.get(0).getUrl().getMethodParameter(methodName, Constants.CLUSTER_STICKY_KEY, Constants.DEFAULT_CLUSTER_STICKY); { //ignore overloaded method // 如果上一次粘滞连接的调用不在可选的提供者列合内,则直接设置为空 if (stickyInvoker != null && !invokers.contains(stickyInvoker)) { stickyInvoker = null; } //ignore concurrency problem // stickyInvoker不为null,并且没在已选列表中,返回上次的服务提供者stickyInvoker,但之前强制校验可达性。 // 由于stickyInvoker不能包含在selected列表中,通过代码看,可以得知forking和failover集群策略,用不了sticky属性 if (sticky && stickyInvoker != null && (selected == null || !selected.contains(stickyInvoker))) { if (availablecheck && stickyInvoker.isAvailable()) { return stickyInvoker; } } } // 利用负载均衡选一个提供者 Invoker invoker = doSelect(loadbalance, invocation, invokers, selected); // 如果启动粘滞连接,则记录这一次的调用 if (sticky) { stickyInvoker = invoker; } return invoker; }
该方法实现了使用负载均衡策略选择一个调用者。首先,使用loadbalance选择一个调用者。如果此调用者位于先前选择的列表中,或者如果此调用者不可用,则重新选择,否则返回第一个选定的调用者。重新选择,重选的验证规则:选择>可用。这条规则可以保证所选的调用者最少有机会成为之前选择的列表中的一个,也是保证这个调用程序可用。
3.doSelectprivate InvokerdoSelect(LoadBalance loadbalance, Invocation invocation, List > invokers, List > selected) throws RpcException { if (invokers == null || invokers.isEmpty()) return null; // 如果只有一个 ,就直接返回这个 if (invokers.size() == 1) return invokers.get(0); // 如果没有指定用哪个负载均衡策略,则默认用随机负载均衡策略 if (loadbalance == null) { loadbalance = ExtensionLoader.getExtensionLoader(LoadBalance.class).getExtension(Constants.DEFAULT_LOADBALANCE); } // 调用负载均衡选择 Invoker invoker = loadbalance.select(invokers, getUrl(), invocation); //If the `invoker` is in the `selected` or invoker is unavailable && availablecheck is true, reselect. // 如果选择的提供者,已在selected中或者不可用则重新选择 if ((selected != null && selected.contains(invoker)) || (!invoker.isAvailable() && getUrl() != null && availablecheck)) { try { // 重新选择 Invoker rinvoker = reselect(loadbalance, invocation, invokers, selected, availablecheck); if (rinvoker != null) { invoker = rinvoker; } else { //Check the index of current selected invoker, if it"s not the last one, choose the one at index+1. // 如果重新选择失败,看下第一次选的位置,如果不是最后,选+1位置. int index = invokers.indexOf(invoker); try { //Avoid collision // 最后再避免选择到同一个invoker invoker = index < invokers.size() - 1 ? invokers.get(index + 1) : invokers.get(0); } catch (Exception e) { logger.warn(e.getMessage() + " may because invokers list dynamic change, ignore.", e); } } } catch (Throwable t) { logger.error("cluster reselect fail reason is :" + t.getMessage() + " if can not solve, you can set cluster.availablecheck=false in url", t); } } return invoker; }
该方法是用负载均衡选择一个invoker的主要逻辑。
4.reselectprivate Invokerreselect(LoadBalance loadbalance, Invocation invocation, List > invokers, List > selected, boolean availablecheck) throws RpcException { //Allocating one in advance, this list is certain to be used. //预先分配一个重选列表,这个列表是一定会用到的. List > reselectInvokers = new ArrayList >(invokers.size() > 1 ? (invokers.size() - 1) : invokers.size()); //First, try picking a invoker not in `selected`. //先从非select中选 //把不包含在selected中的提供者,放入重选列表reselectInvokers,让负载均衡器选择 if (availablecheck) { // invoker.isAvailable() should be checked for (Invoker invoker : invokers) { if (invoker.isAvailable()) { if (selected == null || !selected.contains(invoker)) { reselectInvokers.add(invoker); } } } // 在重选列表中用负载均衡器选择 if (!reselectInvokers.isEmpty()) { return loadbalance.select(reselectInvokers, getUrl(), invocation); } } else { // do not check invoker.isAvailable() // 不核对服务是否可以,把不包含在selected中的提供者,放入重选列表reselectInvokers,让负载均衡器选择 for (Invoker invoker : invokers) { if (selected == null || !selected.contains(invoker)) { reselectInvokers.add(invoker); } } if (!reselectInvokers.isEmpty()) { return loadbalance.select(reselectInvokers, getUrl(), invocation); } } // Just pick an available invoker using loadbalance policy { // 如果非selected的列表中没有选择到,则从selected中选择 if (selected != null) { for (Invoker invoker : selected) { if ((invoker.isAvailable()) // available first && !reselectInvokers.contains(invoker)) { reselectInvokers.add(invoker); } } } if (!reselectInvokers.isEmpty()) { return loadbalance.select(reselectInvokers, getUrl(), invocation); } } return null; }
该方法是是重新选择的逻辑实现。
5.invoke@Override public Result invoke(final Invocation invocation) throws RpcException { // 核对是否已经销毁 checkWhetherDestroyed(); LoadBalance loadbalance = null; // binding attachments into invocation. // 获得上下文的附加值 MapcontextAttachments = RpcContext.getContext().getAttachments(); // 把附加值放入到会话域中 if (contextAttachments != null && contextAttachments.size() != 0) { ((RpcInvocation) invocation).addAttachments(contextAttachments); } // 生成服务提供者集合 List > invokers = list(invocation); if (invokers != null && !invokers.isEmpty()) { // 获得负载均衡器 loadbalance = ExtensionLoader.getExtensionLoader(LoadBalance.class).getExtension(invokers.get(0).getUrl() .getMethodParameter(RpcUtils.getMethodName(invocation), Constants.LOADBALANCE_KEY, Constants.DEFAULT_LOADBALANCE)); } RpcUtils.attachInvocationIdIfAsync(getUrl(), invocation); return doInvoke(invocation, invokers, loadbalance); }
该方法是invoker接口必备的方法,调用链的逻辑,不过主要的逻辑在doInvoke方法中,该方法是该类的抽象方法,让子类只关注doInvoke方法。
6.listprotected List> list(Invocation invocation) throws RpcException { // 把会话域中的invoker加入集合 List > invokers = directory.list(invocation); return invokers; }
该方法是调用了directory的list方法,从会话域中获得所有的Invoker集合。关于directory我会在后续文章讲解。
(二)AvailableClusterpublic class AvailableCluster implements Cluster { public static final String NAME = "available"; @Override publicInvoker join(Directory directory) throws RpcException { // 创建一个AbstractClusterInvoker return new AbstractClusterInvoker (directory) { @Override public Result doInvoke(Invocation invocation, List > invokers, LoadBalance loadbalance) throws RpcException { // 遍历所有的involer,只要有一个可用就直接调用。 for (Invoker invoker : invokers) { if (invoker.isAvailable()) { return invoker.invoke(invocation); } } throw new RpcException("No provider available in " + invokers); } }; } }
Available Cluster我在上面已经讲过了,只要找到一个可用的,则直接调用。
(三)BroadcastClusterpublic class BroadcastCluster implements Cluster { @Override publicInvoker join(Directory directory) throws RpcException { // 创建一个BroadcastClusterInvoker return new BroadcastClusterInvoker (directory); } }
关键实现在于BroadcastClusterInvoker。
(四)BroadcastClusterInvokerpublic class BroadcastClusterInvoker(五)ForkingClusterextends AbstractClusterInvoker { private static final Logger logger = LoggerFactory.getLogger(BroadcastClusterInvoker.class); public BroadcastClusterInvoker(Directory directory) { super(directory); } @Override @SuppressWarnings({"unchecked", "rawtypes"}) public Result doInvoke(final Invocation invocation, List > invokers, LoadBalance loadbalance) throws RpcException { // 检测invokers是否为空 checkInvokers(invokers, invocation); // 把invokers放到上下文 RpcContext.getContext().setInvokers((List) invokers); RpcException exception = null; Result result = null; // 遍历invokers,逐个调用,在循环调用结束后,只要任意一台报错就报错 for (Invoker invoker : invokers) { try { result = invoker.invoke(invocation); } catch (RpcException e) { exception = e; logger.warn(e.getMessage(), e); } catch (Throwable e) { exception = new RpcException(e.getMessage(), e); logger.warn(e.getMessage(), e); } } if (exception != null) { throw exception; } return result; } }
public class ForkingCluster implements Cluster { public final static String NAME = "forking"; @Override public(六)ForkingClusterInvokerInvoker join(Directory directory) throws RpcException { // 创建ForkingClusterInvoker return new ForkingClusterInvoker (directory); } }
public class ForkingClusterInvoker(七)FailbackClusterextends AbstractClusterInvoker { /** * 线程池 * Use {@link NamedInternalThreadFactory} to produce {@link com.alibaba.dubbo.common.threadlocal.InternalThread} * which with the use of {@link com.alibaba.dubbo.common.threadlocal.InternalThreadLocal} in {@link RpcContext}. */ private final ExecutorService executor = Executors.newCachedThreadPool( new NamedInternalThreadFactory("forking-cluster-timer", true)); public ForkingClusterInvoker(Directory directory) { super(directory); } @Override @SuppressWarnings({"unchecked", "rawtypes"}) public Result doInvoke(final Invocation invocation, List > invokers, LoadBalance loadbalance) throws RpcException { try { // 检测invokers是否为空 checkInvokers(invokers, invocation); final List > selected; // 获取 forks 配置 final int forks = getUrl().getParameter(Constants.FORKS_KEY, Constants.DEFAULT_FORKS); // 获取超时配置 final int timeout = getUrl().getParameter(Constants.TIMEOUT_KEY, Constants.DEFAULT_TIMEOUT); // 如果 forks 配置不合理,则直接将 invokers 赋值给 selected if (forks <= 0 || forks >= invokers.size()) { selected = invokers; } else { selected = new ArrayList >(); // 循环选出 forks 个 Invoker,并添加到 selected 中 for (int i = 0; i < forks; i++) { // TODO. Add some comment here, refer chinese version for more details. // 选择 Invoker Invoker invoker = select(loadbalance, invocation, invokers, selected); if (!selected.contains(invoker)) {//Avoid add the same invoker several times. // 加入到selected集合 selected.add(invoker); } } } // 放入上下文 RpcContext.getContext().setInvokers((List) selected); final AtomicInteger count = new AtomicInteger(); final BlockingQueue
public class FailbackCluster implements Cluster { public final static String NAME = "failback"; @Override public(八)FailbackClusterInvoker 1.属性Invoker join(Directory directory) throws RpcException { // 创建一个FailbackClusterInvoker return new FailbackClusterInvoker (directory); } }
private static final Logger logger = LoggerFactory.getLogger(FailbackClusterInvoker.class); // 重试间隔 private static final long RETRY_FAILED_PERIOD = 5 * 1000; /** * 定时器 * Use {@link NamedInternalThreadFactory} to produce {@link com.alibaba.dubbo.common.threadlocal.InternalThread} * which with the use of {@link com.alibaba.dubbo.common.threadlocal.InternalThreadLocal} in {@link RpcContext}. */ private final ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(2, new NamedInternalThreadFactory("failback-cluster-timer", true)); /** * 失败集合 */ private final ConcurrentMap2.doInvoke> failed = new ConcurrentHashMap >(); /** * future */ private volatile ScheduledFuture> retryFuture;
@Override protected Result doInvoke(Invocation invocation, List> invokers, LoadBalance loadbalance) throws RpcException { try { // 检测invokers是否为空 checkInvokers(invokers, invocation); // 选择出invoker Invoker invoker = select(loadbalance, invocation, invokers, null); // 调用 return invoker.invoke(invocation); } catch (Throwable e) { logger.error("Failback to invoke method " + invocation.getMethodName() + ", wait for retry in background. Ignored exception: " + e.getMessage() + ", ", e); // 如果失败,则加入到失败队列,等待重试 addFailed(invocation, this); return new RpcResult(); // ignore } }
该方法是选择invoker调用的逻辑,在抛出异常的时候,做了失败重试的机制,主要实现在addFailed。
3.addFailedprivate void addFailed(Invocation invocation, AbstractClusterInvoker> router) { if (retryFuture == null) { // 锁住 synchronized (this) { if (retryFuture == null) { // 创建定时任务,每隔5秒执行一次 retryFuture = scheduledExecutorService.scheduleWithFixedDelay(new Runnable() { @Override public void run() { // collect retry statistics try { // 对失败的调用进行重试 retryFailed(); } catch (Throwable t) { // Defensive fault tolerance logger.error("Unexpected error occur at collect statistic", t); } } }, RETRY_FAILED_PERIOD, RETRY_FAILED_PERIOD, TimeUnit.MILLISECONDS); } } } // 添加 invocation 和 invoker 到 failed 中 failed.put(invocation, router); }
该方法做的事创建了定时器,然后把失败的调用放入到集合中。
4.retryFailedvoid retryFailed() { // 如果失败队列为0,返回 if (failed.size() == 0) { return; } // 遍历失败队列 for (Map.Entry> entry : new HashMap >( failed).entrySet()) { // 获得会话域 Invocation invocation = entry.getKey(); // 获得invoker Invoker> invoker = entry.getValue(); try { // 重新调用 invoker.invoke(invocation); // 从失败队列中移除 failed.remove(invocation); } catch (Throwable e) { logger.error("Failed retry to invoke method " + invocation.getMethodName() + ", waiting again.", e); } } }
这个方法是调用失败的invoker重新调用的机制。
(九)FailfastClusterpublic class FailfastCluster implements Cluster { public final static String NAME = "failfast"; @Override public(十)FailfastClusterInvokerInvoker join(Directory directory) throws RpcException { // 创建FailfastClusterInvoker return new FailfastClusterInvoker (directory); } }
public class FailfastClusterInvokerextends AbstractClusterInvoker { public FailfastClusterInvoker(Directory directory) { super(directory); } @Override public Result doInvoke(Invocation invocation, List > invokers, LoadBalance loadbalance) throws RpcException { // 检测invokers是否为空 checkInvokers(invokers, invocation); // 选择一个invoker Invoker invoker = select(loadbalance, invocation, invokers, null); try { // 调用 return invoker.invoke(invocation); } catch (Throwable e) { if (e instanceof RpcException && ((RpcException) e).isBiz()) { // biz exception. // 抛出异常 throw (RpcException) e; } // 抛出异常 throw new RpcException(e instanceof RpcException ? ((RpcException) e).getCode() : 0, "Failfast invoke providers " + invoker.getUrl() + " " + loadbalance.getClass().getSimpleName() + " select from all providers " + invokers + " for service " + getInterface().getName() + " method " + invocation.getMethodName() + " on consumer " + NetUtils.getLocalHost() + " use dubbo version " + Version.getVersion() + ", but no luck to perform the invocation. Last error is: " + e.getMessage(), e.getCause() != null ? e.getCause() : e); } } }
逻辑比较简单,调用抛出异常就直接抛出。
(十一)FailoverClusterpublic class FailoverCluster implements Cluster { public final static String NAME = "failover"; @Override public(十二)FailoverClusterInvokerInvoker join(Directory directory) throws RpcException { // 创建FailoverClusterInvoker return new FailoverClusterInvoker (directory); } }
public class FailoverClusterInvokerextends AbstractClusterInvoker { private static final Logger logger = LoggerFactory.getLogger(FailoverClusterInvoker.class); public FailoverClusterInvoker(Directory directory) { super(directory); } @Override @SuppressWarnings({"unchecked", "rawtypes"}) public Result doInvoke(Invocation invocation, final List > invokers, LoadBalance loadbalance) throws RpcException { // 复制一个invoker集合 List > copyinvokers = invokers; // 检测是否为空 checkInvokers(copyinvokers, invocation); // 获取重试次数 int len = getUrl().getMethodParameter(invocation.getMethodName(), Constants.RETRIES_KEY, Constants.DEFAULT_RETRIES) + 1; if (len <= 0) { len = 1; } // retry loop. // 记录最后一个异常 RpcException le = null; // last exception. List > invoked = new ArrayList >(copyinvokers.size()); // invoked invokers. Set providers = new HashSet (len); // 循环调用,失败重试 for (int i = 0; i < len; i++) { //Reselect before retry to avoid a change of candidate `invokers`. //NOTE: if `invokers` changed, then `invoked` also lose accuracy. // 在进行重试前重新列举 Invoker,这样做的好处是,如果某个服务挂了, // 通过调用 list 可得到最新可用的 Invoker 列表 if (i > 0) { checkWhetherDestroyed(); copyinvokers = list(invocation); // check again // 检测copyinvokers 是否为空 checkInvokers(copyinvokers, invocation); } // 通过负载均衡选择invoker Invoker invoker = select(loadbalance, invocation, copyinvokers, invoked); // 添加到 invoker 到 invoked 列表中 invoked.add(invoker); // 设置 invoked 到 RPC 上下文中 RpcContext.getContext().setInvokers((List) invoked); try { // 调用目标 Invoker 的 invoke 方法 Result result = invoker.invoke(invocation); if (le != null && logger.isWarnEnabled()) { logger.warn("Although retry the method " + invocation.getMethodName() + " in the service " + getInterface().getName() + " was successful by the provider " + invoker.getUrl().getAddress() + ", but there have been failed providers " + providers + " (" + providers.size() + "/" + copyinvokers.size() + ") from the registry " + directory.getUrl().getAddress() + " on the consumer " + NetUtils.getLocalHost() + " using the dubbo version " + Version.getVersion() + ". Last error is: " + le.getMessage(), le); } return result; } catch (RpcException e) { if (e.isBiz()) { // biz exception. throw e; } le = e; } catch (Throwable e) { le = new RpcException(e.getMessage(), e); } finally { providers.add(invoker.getUrl().getAddress()); } } // 若重试失败,则抛出异常 throw new RpcException(le != null ? le.getCode() : 0, "Failed to invoke the method " + invocation.getMethodName() + " in the service " + getInterface().getName() + ". Tried " + len + " times of the providers " + providers + " (" + providers.size() + "/" + copyinvokers.size() + ") from the registry " + directory.getUrl().getAddress() + " on the consumer " + NetUtils.getLocalHost() + " using the dubbo version " + Version.getVersion() + ". Last error is: " + (le != null ? le.getMessage() : ""), le != null && le.getCause() != null ? le.getCause() : le); } }
该类实现了失败重试的容错策略,当调用失败的时候,记录下异常,然后循环调用下一个选择出来的invoker,直到重试次数用完,抛出最后一次的异常。
(十三)FailsafeClusterpublic class FailsafeCluster implements Cluster { public final static String NAME = "failsafe"; @Override public(十四)FailsafeClusterInvokerInvoker join(Directory directory) throws RpcException { // 创建FailsafeClusterInvoker return new FailsafeClusterInvoker (directory); } }
public class FailsafeClusterInvokerextends AbstractClusterInvoker { private static final Logger logger = LoggerFactory.getLogger(FailsafeClusterInvoker.class); public FailsafeClusterInvoker(Directory directory) { super(directory); } @Override public Result doInvoke(Invocation invocation, List > invokers, LoadBalance loadbalance) throws RpcException { try { // 检测invokers是否为空 checkInvokers(invokers, invocation); // 选择一个invoker Invoker invoker = select(loadbalance, invocation, invokers, null); // 调用 return invoker.invoke(invocation); } catch (Throwable e) { // 如果失败打印异常,返回一个空结果 logger.error("Failsafe ignore exception: " + e.getMessage(), e); return new RpcResult(); // ignore } } }
逻辑比较简单,就是不抛出异常,只是打印异常。
后记该部分相关的源码解析地址:https://github.com/CrazyHZM/i...
该文章讲解了集群中关于cluster实现的部分,讲了几种调用方式和容错策略。接下来我将开始对集群模块关于配置规则部分进行讲解。
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/77383.html
摘要:可以参考源码解析二十四远程调用协议的八。十六的该类也是用了适配器模式,该类主要的作用就是增加了心跳功能,可以参考源码解析十远程通信层的四。二十的可以参考源码解析十七远程通信的一。 2.7大揭秘——消费端发送请求过程 目标:从源码的角度分析一个服务方法调用经历怎么样的磨难以后到达服务端。 前言 前一篇文章讲到的是引用服务的过程,引用服务无非就是创建出一个代理。供消费者调用服务的相关方法。...
摘要:源码分析一创建该类实现了接口,是分组集合的集群实现。三工厂类,获得指定类型的对象。后记该部分相关的源码解析地址该文章讲解了集群中关于分组聚合实现的部分。接下来我将开始对集群模块关于路由部分进行讲解。 集群——merger 目标:介绍dubbo中集群的分组聚合,介绍dubbo-cluster下merger包的源码。 前言 按组合并返回结果 ,比如菜单服务,接口一样,但有多种实现,用gro...
摘要:服务引用过程目标从源码的角度分析服务引用过程。并保留服务提供者的部分配置,比如版本,,时间戳等最后将合并后的配置设置为查询字符串中。的可以参考源码解析二十三远程调用的一的源码分析。 dubbo服务引用过程 目标:从源码的角度分析服务引用过程。 前言 前面服务暴露过程的文章讲解到,服务引用有两种方式,一种就是直连,也就是直接指定服务的地址来进行引用,这种方式更多的时候被用来做服务测试,不...
摘要:集群目标介绍中集群的负载均衡,介绍下包的源码。源码分析一该类实现了接口,是负载均衡的抽象类,提供了权重计算的功能。四该类是负载均衡基于一致性的逻辑实现。 集群——LoadBalance 目标:介绍dubbo中集群的负载均衡,介绍dubbo-cluster下loadBalance包的源码。 前言 负载均衡,说的通俗点就是要一碗水端平。在这个时代,公平是很重要的,在网络请求的时候同样是这个...
摘要:首先将根据路由规则服务提供者和配置规则三种类型分开,分别放入三个集合,然后对每个集合进行修改或者通知设置禁止访问置空关闭所有的关闭禁止访问引用老的传入的为空,说明是路由规则或配置规则发生改变,此时是空的,直接使用。 集群——directory 目标:介绍dubbo中集群的目录,介绍dubbo-cluster下directory包的源码。 前言 我在前面的文章中也提到了Directory...
阅读 3448·2023-04-26 00:39
阅读 4038·2021-09-22 10:02
阅读 2532·2021-08-09 13:46
阅读 1098·2019-08-29 18:40
阅读 1443·2019-08-29 18:33
阅读 773·2019-08-29 17:14
阅读 1513·2019-08-29 12:40
阅读 2969·2019-08-28 18:07