Dubbo中关于服务的订阅和通知主要发生在服务提供方暴露服务的过程和服务消费方初始化时候引用服务的过程中。
服务引用过程中的订阅和通知
在服务消费者初始化的过程中,会有一步是进行服务的引用,具体的代码是在RegistryProtocol的refer方法:
1 |
public <T> Invoker<T> refer(Class<T> type, URL url) throws RpcException { |
在refer方法中有一步是获取注册中心实例,这一步中也会有一个notify操作,先暂时不解释。接着就是doRefer方法:
1 |
private <T> Invoker<T> doRefer(Cluster cluster, Registry registry, Class<T> type, URL url) { |
在doRefer方法中服务消费者会订阅服务,同时订阅了三种类型:providers,routers,configurators。
接续看directory.subscribe订阅方法,这里directory是RegistryDirectory:
1 |
public void subscribe(URL url) { |
这里registry是ZookeeperRegistry,在ZookeeperRegistry调用subscribe处理之前会先经过AbstractRegistry的处理,然后经过FailbackRegistry处理,在FailbackRegistry中会调用ZookeeperRegistry的doSubscribe方法。
首先看下AbstractRegistry中subscribe方法:
1 |
public void subscribe(URL url, NotifyListener listener) { |
然后是FailbackRegistry的subscribe方法:
1 |
public void subscribe(URL url, NotifyListener listener) { |
这里总共进行了一下几件事情:
- AbstractRegistry的处理
- 移除订阅失败的
- 由具体的子类向服务器端发送订阅请求
- 如果订阅发生失败了,尝试获取缓存url,然后进行失败通知或者如果开启了启动时检测,则直接抛出异常
- 将失败的订阅请求记录到失败列表,定时重试
主要看下子类向服务器段发送订阅请求的步骤,在ZookeeperRegistry的doSubscribe方法中:
1 |
protected void doSubscribe(final URL url, final NotifyListener listener) { |
上面主要是分别对providers,routers,configurators三种不同类型的进行订阅,也就是往zookeeper中注册节点,注册之前先给url添加监听器。最后是订阅完之后进行通知。
notify方法,这里notify方法实现是在ZookeeperRegistry的父类FailbackRegistry中:
1 |
protected void notify(URL url, NotifyListener listener, List<URL> urls) { |
看下AbstractRegistry的notify方法:
1 |
protected void notify(URL url, NotifyListener listener, List<URL> urls) { |
继续看RegistryDirectory的notify方法:
1 |
public synchronized void notify(List<URL> urls) { |
最重要的重建invoker实例,在服务引用的文章中已经介绍过,不再重复,还有上面说省略的获取注册中心实例的过程中,也会有notify的操作。(这里省略)
这里也是进行了invoker实例的重建。
暴露服务过程中的订阅和通知
服务暴露过程中的订阅在RegistryProtocol的export方法中:
1 |
public <T> Exporter<T> export(final Invoker<T> originInvoker) throws RpcException { |
registry.subscribe订阅override数据,会首先经过AbstractRegistry处理,然后经过FailbackRegistry处理。处理方法在上面消费者发布订阅的讲解中都已经介绍。往下的步骤基本相同,不同之处在于AbstractRegistry的notify方法:
1 |
protected void notify(URL url, NotifyListener listener, List<URL> urls) { |
接下来看OverrideListener的notify方法:
1 |
/* |
这里也是对Invoker重新进行了引用。