4.dubbo源碼-發(fā)布服務(wù)

dubbo服務(wù)發(fā)布

dubbo服務(wù)發(fā)布只需在spring.xml中如下配置即可:
<dubbo:service interface="com.alibaba.dubbo.demo.DemoService" ref="demoService" />

export初始化

通過2-dubbo結(jié)合spring可知,<dubbo:service>解析后封裝到ServiceBean中;ServiceBean定義如下,繼承了dubbo定義的類ServiceConfig,實(shí)現(xiàn)了5個(gè)spring的接口,為了融入spring容器的啟動(dòng)過程中:

public class ServiceBean<T> extends ServiceConfig<T> implements InitializingBean, DisposableBean, ApplicationContextAware, ApplicationListener, BeanNameAware {
    ... ...
}

ServiceBean實(shí)現(xiàn)了ApplicationListener接口,當(dāng)spring容器觸發(fā)了ContextRefreshedEvent事件時(shí),就會(huì)調(diào)用ServiceConfig中的export()方法發(fā)布<dubbo:service>申明的dubbo服務(wù),且在dubbo的info級(jí)別日志中有相應(yīng)的日志:

public void onApplicationEvent(ApplicationEvent event) {
    if (ContextRefreshedEvent.class.getName().equals(event.getClass().getName())) {
        if (isDelay() && ! isExported() && ! isUnexported()) {
            if (logger.isInfoEnabled()) {
                logger.info("The service ready on spring started. service: " + getInterface());
            }
            export();
        }
    }
}

info日志示例:The service ready on spring started. service: com.alibaba.dubbo.demo.DemoService

ServiceConfig.export()

ServiceConfigexport()方法部分源碼如下,如果<dubbo:service>中申明了delay(例如<dubbo:service interface="com.alibaba.dubbo.demo.TestService" ref="testService" delay="3000"/>),那么延遲調(diào)用doExport()發(fā)布這個(gè)服務(wù),否則直接調(diào)用doExport()發(fā)布服務(wù):

public synchronized void export() {
    ... ...
    if (delay != null && delay > 0) {
        Thread thread = new Thread(new Runnable() {
            public void run() {
                try {
                    Thread.sleep(delay);
                } catch (Throwable e) {
                }
                doExport();
            }
        });
        thread.setDaemon(true);
        thread.setName("DelayExportServiceThread");
        thread.start();
    } else {
        doExport();
    }
}

ServiceConfig.doExport()的作用:

  1. 檢查<dubbo:service>中是否配置了interface, 如果為空,那么拋出異常:
  2. 檢查xml配置中申明的interface的類型是否是java interface類型(interfaceClass.isInterface())
  3. 檢查xml配置中interface和ref是否匹配(interfaceClass.isInstance(ref))
  4. application&registry&protocol等有效性檢查;
  5. 有效性檢查通過后,調(diào)用doExportUrls()發(fā)布dubbo服務(wù);

ServiceConfig.doExportUrls()

通過調(diào)用loadRegistries(true)得到所有registry的url地址,例如在dubbo.properties中通過配置dubbo.registry.address=zookeeper://127.0.0.1:2181;protocols就是將要發(fā)布服務(wù)的協(xié)議集合(dubbo服務(wù)可以同時(shí)暴露多種協(xié)議),可以在dubbo.properties中配置,以dubbo協(xié)議為例:

dubbo.protocol.name=dubbo
dubbo.protocol.port=20880

ServiceConfig.doExportUrls()源碼如下:

private void doExportUrls() {
    List<URL> registryURLs = loadRegistries(true);
    // 一般只配置dubbo協(xié)議,那么protocols就是:<dubbo:protocol name="dubbo" port="20880" id="dubbo" />
    for (ProtocolConfig protocolConfig : protocols) {
        doExportUrlsFor1Protocol(protocolConfig, registryURLs);
    }
}

ServiceConfig.doExportUrlsFor1Protocol()

先把所有相關(guān)屬性封裝到Map中,例如protocol=dubbo,host=10.0.0.1,port=20880,path=com.alibaba.dubbo.demo.TestService等,然后構(gòu)造dubbo定義的統(tǒng)一數(shù)據(jù)模型URL:

URL url = new URL(name, host, port, (contextPath == null || contextPath.length() == 0 ? "" : contextPath + "/") + path, map);

得到的url如下所示(這個(gè)url非常重要,貫穿整個(gè)dubbo服務(wù)的發(fā)布和調(diào)用過程,可以在服務(wù)發(fā)布后在dubbo-monitor中看到):

ServiceConfig.doExportUrlsFor1Protocol()中根據(jù)scope判斷服務(wù)的發(fā)布范圍:

  1. 如果配置scope=none, 那么不需要發(fā)布這個(gè)dubbo服務(wù);
  2. 沒有配置scope=noe且配置的scope!=remote, 那么本地暴露 這個(gè)dubbo服務(wù);
  3. 沒有配置scope=noe且配置的scope!=remote且配置的scope!=local,那么遠(yuǎn)程暴露這個(gè)dubbo服務(wù)(例如遠(yuǎn)程暴露這個(gè)服務(wù)到zk上,默認(rèn)情況下scope沒有配置,就是在這里發(fā)布服務(wù));

實(shí)現(xiàn)源碼如下:

//配置為none不暴露
if (! Constants.SCOPE_NONE.toString().equalsIgnoreCase(scope)) {

    //配置不是remote的情況下做本地暴露 (配置為remote,則表示只暴露遠(yuǎn)程服務(wù))
    if (!Constants.SCOPE_REMOTE.toString().equalsIgnoreCase(scope)) {
        exportLocal(url);
    }
    //如果配置不是local則暴露為遠(yuǎn)程服務(wù).(配置為local,則表示只暴露遠(yuǎn)程服務(wù))
    if (! Constants.SCOPE_LOCAL.toString().equalsIgnoreCase(scope) ){
        if (logger.isInfoEnabled()) {
            logger.info("Export dubbo service " + interfaceClass.getName() + " to url " + url);
        }
        // 如果注冊(cè)u(píng)rl地址存在,例如申明了注冊(cè)的zk地址
        if (registryURLs != null && registryURLs.size() > 0
                && url.getParameter("register", true)) {
            // 注冊(cè)的zk地址可能是集群,那么需要遍歷這些地址一一進(jìn)行注冊(cè)
            for (URL registryURL : registryURLs) {
                url = url.addParameterIfAbsent("dynamic", registryURL.getParameter("dynamic"));
                // 如果申明了dubbo-monitor,那么再url地址上append類似monitor=monitor全地址
                URL monitorUrl = loadMonitor(registryURL);
                if (monitorUrl != null) {
                    url = url.addParameterAndEncoded(Constants.MONITOR_KEY, monitorUrl.toFullString());
                }
                if (logger.isInfoEnabled()) {
                    logger.info("Register dubbo service " + interfaceClass.getName() + " url " + url + " to registry " + registryURL);
                }
                Invoker<?> invoker = proxyFactory.getInvoker(ref, (Class) interfaceClass, registryURL.addParameterAndEncoded(Constants.EXPORT_KEY, url.toFullString()));
                // 默認(rèn)都是dubbo協(xié)議,所以調(diào)用DubboProtol.export(Invoker)
                Exporter<?> exporter = protocol.export(invoker);
                exporters.add(exporter);
            }
        } else {
            Invoker<?> invoker = proxyFactory.getInvoker(ref, (Class) interfaceClass, url);
            Exporter<?> exporter = protocol.export(invoker);
            exporters.add(exporter);
        }
    }
}

由這段源碼可知,如果發(fā)布dubbo服務(wù)到zookeeper上,invoker.getUrl()的值為:
registry://10.0.53.87:2188/com.alibaba.dubbo.registry.RegistryService?application=dubbo-test&dubbo=2.0.0&export=dubbo%3A%2F%2F10.52.16.218%3A20886%2Fcom.alibaba.dubbo.demo.DemoService%3Fanyhost%3Dtrue%26application%3Ddubbo-test%26dubbo%3D2.0.0%26interface%3Dcom.alibaba.dubbo.demo.DemoService%26loadbalance%3Droundrobin%26methods%3DsayHello%26owner%3Dafei%26pid%3D2380%26side%3Dprovider%26timestamp%3D1509953019382&owner=afei&pid=2380&registry=zookeeper&timestamp=1509953019349

且會(huì)有兩行info級(jí)別的日志;

Export dubbo service com.alibaba.dubbo.demo.DemoService to url ... ...
Register dubbo service com.alibaba.dubbo.demo.DemoService url ... ...

Protocol.export()

com.alibaba.dubbo.rpc.Protocol中暴露服務(wù)接口申明:

/**
 * 暴露遠(yuǎn)程服務(wù):<br>
 * 1. 協(xié)議在接收請(qǐng)求時(shí),應(yīng)記錄請(qǐng)求來源方地址信息:RpcContext.getContext().setRemoteAddress();<br>
 * 2. export()必須是冪等的,也就是暴露同一個(gè)URL的Invoker兩次,和暴露一次沒有區(qū)別。<br>
 * 3. export()傳入的Invoker由框架實(shí)現(xiàn)并傳入,協(xié)議不需要關(guān)心。<br>
 * 
 * @param <T> 服務(wù)的類型
 * @param invoker 服務(wù)的執(zhí)行體
 * @return exporter 暴露服務(wù)的引用,用于取消暴露
 * @throws RpcException 當(dāng)暴露服務(wù)出錯(cuò)時(shí)拋出,比如端口已占用
 */
@Adaptive
<T> Exporter<T> export(Invoker<T> invoker) throws RpcException;

RegistryProtocol.export()

源碼如下:

public <T> Exporter<T> export(final Invoker<T> originInvoker) throws RpcException {
    //export invoker
    final ExporterChangeableWrapper<T> exporter = doLocalExport(originInvoker);
    //registry provider,根據(jù)發(fā)布的服務(wù)originInvoker得到Registry實(shí)例,由于一般都使用zookeeper為注冊(cè)中心,所以這里得到的是ZookeeperRegistry;
    final Registry registry = getRegistry(originInvoker);
    final URL registedProviderUrl = getRegistedProviderUrl(originInvoker);
    // 所以這里調(diào)用ZookeeperRegistry.register(URL)把需要發(fā)布的服務(wù)注冊(cè)到zookeeper中
    registry.register(registedProviderUrl);
    // 訂閱override數(shù)據(jù)
    // FIXME 提供者訂閱時(shí),會(huì)影響同一JVM即暴露服務(wù),又引用同一服務(wù)的的場(chǎng)景,因?yàn)閟ubscribed以服務(wù)名為緩存的key,導(dǎo)致訂閱信息覆蓋。
    final URL overrideSubscribeUrl = getSubscribedOverrideUrl(registedProviderUrl);
    final OverrideListener overrideSubscribeListener = new OverrideListener(overrideSubscribeUrl);
    overrideListeners.put(overrideSubscribeUrl, overrideSubscribeListener);
    registry.subscribe(overrideSubscribeUrl, overrideSubscribeListener);
    //保證每次export都返回一個(gè)新的exporter實(shí)例
    return new Exporter<T>() {
        public Invoker<T> getInvoker() {
            return exporter.getInvoker();
        }
        public void unexport() {
            try {
                exporter.unexport();
            } catch (Throwable t) {
                logger.warn(t.getMessage(), t);
            }
            try {
                registry.unregister(registedProviderUrl);
            } catch (Throwable t) {
                logger.warn(t.getMessage(), t);
            }
            try {
                overrideListeners.remove(overrideSubscribeUrl);
                registry.unsubscribe(overrideSubscribeUrl, overrideSubscribeListener);
            } catch (Throwable t) {
                logger.warn(t.getMessage(), t);
            }
        }
    };
}

核心調(diào)用registry.register(registedProviderUrl)

  1. 調(diào)用AbstractRegistry.register(URL),把這次需要注冊(cè)的URL加到Set<URL> registered中,即本地緩存新的注冊(cè)URL;
  2. 在ZookeeperRegistry.doRegister(URL)調(diào)用AbstractZookeeperClient.create(),toUrlPath將URL形式的地址轉(zhuǎn)換成zookeeper路徑,最終在AbstractZookeeperClient中把需要發(fā)布的服務(wù)的URL保存到zookeeper中;(依賴第三方j(luò)ar包:org.I0Itec.zkclient)
  3. ZookeeperRegistry.doRegister(url)注冊(cè)服務(wù)如果失敗:
  • 如果開啟了啟動(dòng)檢查check=true,那么直接拋出異常;
  • 如果沒有開啟啟動(dòng)檢查,那么將失敗的注冊(cè)請(qǐng)求記錄到失敗列表,定時(shí)重試;

核心調(diào)用registry.subscribe(overrideSubscribeUrl, overrideSubscribeListener):

  1. 對(duì)發(fā)布的dubbo服務(wù)的這個(gè)url進(jìn)行監(jiān)聽, 當(dāng)服務(wù)變化有時(shí)通知重新暴露服務(wù), 以zookeeper為例,暴露服務(wù)會(huì)在zookeeper生成一個(gè)節(jié)點(diǎn),當(dāng)節(jié)點(diǎn)發(fā)生變化的時(shí)候會(huì)觸發(fā)overrideSubscribeListener的notify方法重新暴露服務(wù);

重試機(jī)制

注冊(cè)服務(wù)失敗后,會(huì)將url加入重試url集合中,failedRegistered.add(url); 重試任務(wù)在FailbackRegistry中實(shí)現(xiàn):

public FailbackRegistry(URL url) {
    super(url);
    int retryPeriod = url.getParameter(Constants.REGISTRY_RETRY_PERIOD_KEY, Constants.DEFAULT_REGISTRY_RETRY_PERIOD);
    // retryExecutor是一個(gè)單獨(dú)的線程池Executors.newScheduledThreadPool(1, new NamedThreadFactory("DubboRegistryFailedRetryTimer", true)); 默認(rèn)重試周期是5s;
    this.retryFuture = retryExecutor.scheduleWithFixedDelay(new Runnable() {
        public void run() {
            // 檢測(cè)并連接注冊(cè)中心
            try {
                retry();
            } catch (Throwable t) { // 防御性容錯(cuò)
                logger.error("Unexpected error occur at failed retry, cause: " + t.getMessage(), t);
            }
        }
    }, retryPeriod, retryPeriod, TimeUnit.MILLISECONDS);
}

監(jiān)聽機(jī)制

  1. 訂閱并設(shè)置監(jiān)聽registry.subscribe(overrideSubscribeUrl, overrideSubscribeListener);
    --> FailbackRegistry.subscribe(URL url, NotifyListener listener)
    --> ZookeeperRegistry.doSubscribe(final URL url, final NotifyListener listener),部分實(shí)現(xiàn)源碼如下:
ChildListener zkListener = listeners.get(listener);
if (zkListener == null) {
    listeners.putIfAbsent(listener, new ChildListener() {
        public void childChanged(String parentPath, List<String> currentChilds) {
            ZookeeperRegistry.this.notify(url, listener, toUrlsWithEmpty(url, parentPath, currentChilds));
        }
    });
    zkListener = listeners.get(listener);
}
zkClient.create(path, false);
// 在準(zhǔn)備監(jiān)聽的path上添加ChildListener
List<String> children = zkClient.addChildListener(path, zkListener);
  1. 服務(wù)有變化時(shí)notify:
    FailbackRegistry.notify(URL url, NotifyListener listener, List<URL> urls)
    以path:/dubbo/com.alibaba.dubbo.demo.TestService/providers為例,Consumer會(huì)監(jiān)聽這個(gè)zk路徑;
  • 假設(shè)在consumer啟動(dòng)時(shí)只有1個(gè)provider:dubbo://10.52.17.98:20888... ;當(dāng)再啟動(dòng)一個(gè)provider:dubbo://10.52.17.98:20886后,path路徑/dubbo/com.alibaba.dubbo.demo.TestService/providers就會(huì)變化,結(jié)果就觸發(fā)notify(URL url, NotifyListener listener, List<URL> urls),此時(shí)List<URL> urls中有兩個(gè)provider,即dubbo://10.52.17.98:20886...和dubbo://10.52.17.98:20888...
  • 或者在consumer啟動(dòng)時(shí)有2個(gè)provider:dubbo://10.52.17.98:20886... 和 dubbo://10.52.17.98:20888... ;當(dāng)關(guān)閉一個(gè)provider:dubbo://10.52.17.98:20886后,path路徑/dubbo/com.alibaba.dubbo.demo.TestService/providers也會(huì)變化,結(jié)果就觸發(fā)notify(URL url, NotifyListener listener, List<URL> urls),此時(shí)List<URL> urls中只有1個(gè)provider,即dubbo://10.52.17.98:20888...;
    --> AbstractRegistry.notify(URL url, NotifyListener listener, List<URL> urls)
    --> RegistryDirectory.notify(List<URL> urls)
    --> RegistryDirectory.refreshInvoker(List<URL> invokerUrls),這里調(diào)用toMethodInvokers(Map<String, Invoker<T>> invokersMap)的實(shí)現(xiàn)比較重要,將invokers列表轉(zhuǎn)成與方法的映射關(guān)系,且每個(gè)方法對(duì)應(yīng)的List<Invoker>需要通過Collections.sort(methodInvokers, InvokerComparator.getComparator());排序,排序后,還要將其轉(zhuǎn)為unmodifiable的map:
for (String method : new HashSet<String>(newMethodInvokerMap.keySet())) {
    List<Invoker<T>> methodInvokers = newMethodInvokerMap.get(method);
    Collections.sort(methodInvokers, InvokerComparator.getComparator());
    newMethodInvokerMap.put(method, Collections.unmodifiableList(methodInvokers));
}
return Collections.unmodifiableMap(newMethodInvokerMap);

其中InvokerComparator 的定義如下,即直接根據(jù)url進(jìn)行比較排序:

private static class InvokerComparator implements Comparator<Invoker<?>> {   
    private static final InvokerComparator comparator = new InvokerComparator();  
    public static InvokerComparator getComparator() {
        return comparator;
    }   
    private InvokerComparator() {}
    public int compare(Invoker<?> o1, Invoker<?> o2) {
        return o1.getUrl().toString().compareTo(o2.getUrl().toString());
    }
}

最后刷新本地緩存的方法和List<Invoker>關(guān)系:
this.methodInvokerMap = multiGroup ? toMergeMethodInvokerMap(newMethodInvokerMap) : newMethodInvokerMap;

DubboProtocol.export()

dubbo協(xié)議發(fā)布服務(wù)會(huì)調(diào)用DubboProtocol.export(),

  1. 從Invoker中獲取URL:
    URL url = invoker.getUrl();
  2. 根據(jù)URL得到key, 由暴露的服務(wù)接口+端口組成,例如com.alibaba.dubbo.demo.DemoService:20886
    String key = serviceKey(url);
  3. 構(gòu)造DubboExporter存到Map中l(wèi)ocal cache化:
    DubboExporter<T> exporter = new DubboExporter<T>(invoker, key, exporterMap);
    exporterMap.put(key, exporter);
  4. 調(diào)用DubboProtocol.openServer()開啟netty(默認(rèn))服務(wù)保持通信,并設(shè)置requestHandler處理consumer對(duì)provider的調(diào)用請(qǐng)求;

DubboProtocol.openServer():
key的值就是IP:Port,例如10.52.17.167:20886,根據(jù)key從serverMap中如果取不到ExchangeServer,表示還沒綁定服務(wù)端口,需要調(diào)用createServer(url)-->Exchangers.bind(url, requestHandler)-->Transporters.getTransporter().bind(url, handler)(dubbo支持mina,netty,grizzly,默認(rèn)實(shí)現(xiàn)是netty) --> NettyTransporter.bind(URL, ChannelHandler) --> NettyServer.open();

NettyServer.open()源碼如下:

@Override
rotected void doOpen() throws Throwable {
    NettyHelper.setNettyLoggerFactory();
    ExecutorService boss = Executors.newCachedThreadPool(new NamedThreadFactory("NettyServerBoss", true));
    ExecutorService worker = Executors.newCachedThreadPool(new NamedThreadFactory("NettyServerWorker", true));
    ChannelFactory channelFactory = new NioServerSocketChannelFactory(boss, worker, getUrl().getPositiveParameter(Constants.IO_THREADS_KEY, Constants.DEFAULT_IO_THREADS));
    bootstrap = new ServerBootstrap(channelFactory);
    
    final NettyHandler nettyHandler = new NettyHandler(getUrl(), this);
    channels = nettyHandler.getChannels();
    // https://issues.jboss.org/browse/NETTY-365
    // https://issues.jboss.org/browse/NETTY-379
    // final Timer timer = new HashedWheelTimer(new NamedThreadFactory("NettyIdleTimer", true));
    bootstrap.setPipelineFactory(new ChannelPipelineFactory() {
        public ChannelPipeline getPipeline() {
            NettyCodecAdapter adapter = new NettyCodecAdapter(getCodec() ,getUrl(), NettyServer.this);
            ChannelPipeline pipeline = Channels.pipeline();
            /*int idleTimeout = getIdleTimeout();
            if (idleTimeout > 10000) {
                pipeline.addLast("timer", new IdleStateHandler(timer, idleTimeout / 1000, 0, 0));
            }*/
            pipeline.addLast("decoder", adapter.getDecoder());
            pipeline.addLast("encoder", adapter.getEncoder());
            pipeline.addLast("handler", nettyHandler);
            return pipeline;
        }
    });
    // bind
    channel = bootstrap.bind(getBindAddress());
}

開啟Netty服務(wù)幾個(gè)重要的地方

  1. 構(gòu)造ChannelPipeline時(shí)指定了編碼&解碼,其中編碼為NettyCodecAdapter.getEncoder(),解碼為NettyCodecAdapter.getDncoder();
  2. 指定了handler為final NettyHandler nettyHandler = new NettyHandler(getUrl(), this);處理請(qǐng)求;

附dubbo官方給出的暴露服務(wù)時(shí)序圖:
https://dubbo.gitbooks.io/dubbo-dev-book/design.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,505評(píng)論 6 533
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,556評(píng)論 3 418
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,463評(píng)論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,009評(píng)論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,778評(píng)論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,218評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,281評(píng)論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,436評(píng)論 0 288
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,969評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,795評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,993評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,537評(píng)論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,229評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,659評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,917評(píng)論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,687評(píng)論 3 392
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,990評(píng)論 2 374

推薦閱讀更多精彩內(nèi)容

  • dubbo暴露服務(wù)有兩種情況,一種是設(shè)置了延遲暴露(比如delay="5000"),另外一種是沒有設(shè)置延遲暴露或者...
    加大裝益達(dá)閱讀 21,296評(píng)論 5 36
  • 0 準(zhǔn)備 安裝注冊(cè)中心:Zookeeper、Dubbox自帶的dubbo-registry-simple;安裝Du...
    七寸知架構(gòu)閱讀 14,003評(píng)論 0 88
  • Dubbo是什么 Dubbo是Alibaba開源的分布式服務(wù)框架,它最大的特點(diǎn)是按照分層的方式來架構(gòu),使用這種方式...
    Coselding閱讀 17,253評(píng)論 3 196
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,785評(píng)論 18 139
  • 喜氣洋洋的年, 一轉(zhuǎn)眼,就過去了。 雖然是國家規(guī)定的節(jié)假日,休息過后卻覺得無比的勞累。 與放假前對(duì)于節(jié)日的期盼形成...
    炅竹春筍閱讀 236評(píng)論 0 0