使用Dubbo中需要注意的事項(xiàng)

一坑鱼、前言

Dubbo作為高性能RPC框架膘流,已經(jīng)進(jìn)入Apache卵化器項(xiàng)目,雖然官方給出了dubbo使用的用戶手冊(cè)鲁沥,但是大多是一概而過呼股,使用dubbo時(shí)候要盡量了解源碼,不然會(huì)很容易入坑画恰。

二 彭谁、服務(wù)消費(fèi)端ReferenceConfig需要自行緩存

ReferenceConfig實(shí)例是個(gè)很重的實(shí)例,每個(gè)ReferenceConfig實(shí)例里面都維護(hù)了與服務(wù)注冊(cè)中心的一個(gè)長(zhǎng)鏈允扇,并且維護(hù)了與所有服務(wù)提供者的的長(zhǎng)鏈缠局。假設(shè)有一個(gè)服務(wù)注冊(cè)中心和N個(gè)服務(wù)提供者,那么每個(gè)ReferenceConfig實(shí)例里面維護(hù)了N+1個(gè)長(zhǎng)鏈考润,如果頻繁的生成ReferenceConfig實(shí)例狭园,可能會(huì)造成性能問題,甚至產(chǎn)生內(nèi)存或者連接泄露的風(fēng)險(xiǎn)糊治。特別是使用dubbo api編程時(shí)候容易忽略這個(gè)問題唱矛。

為了解決這個(gè)問題,之前都是自行緩存井辜,但是自從dubbo2.4.0版本后揖赴,dubbo 提供了簡(jiǎn)單的工具類 ReferenceConfigCache 用于緩存ReferenceConfig 實(shí)例。使用如下:

//創(chuàng)建服務(wù)消費(fèi)實(shí)例
ReferenceConfig<XxxService> reference = new ReferenceConfig<XxxService>();
reference.setInterface(XxxService.class);
reference.setVersion("1.0.0");
......
//獲取dubbo提供的緩存
ReferenceConfigCache cache = ReferenceConfigCache.getCache();
// cache.get方法中會(huì)緩存 reference對(duì)象抑胎,并且調(diào)用reference.get方法啟動(dòng)ReferenceConfig,并返回經(jīng)過代理后的服務(wù)接口的對(duì)象
XxxService xxxService = cache.get(reference);

// 使用xxxService對(duì)象
xxxService.sayHello();

需要注意的是 Cache內(nèi)持有ReferenceConfig對(duì)象的引用渐北,不要在外部再調(diào)用ReferenceConfig的destroy方法了阿逃,這會(huì)導(dǎo)致Cache內(nèi)的ReferenceConfig失效!

如果要銷毀 Cache 中的 ReferenceConfig ,將銷毀 ReferenceConfig 并釋放對(duì)應(yīng)的資源恃锉,具體使用下面方法來銷毀:

ReferenceConfigCache cache = ReferenceConfigCache.getCache()搀菩;
cache.destroy(reference);

另外以服務(wù) Group、接口破托、版本為緩存的 Key肪跋,ReferenceConfig實(shí)例為對(duì)應(yīng)的value。如果你需要使用自定義的key土砂,可以在創(chuàng)建cache時(shí)候調(diào)用ReferenceConfigCache cache = ReferenceConfigCache.getCache(keyGenerator );方法傳遞自定義的keyGenerator州既。

三、 并發(fā)控制

3.1 服務(wù)消費(fèi)方并發(fā)控制

在服務(wù)消費(fèi)方法進(jìn)行并發(fā)控制需要設(shè)置actives參數(shù)萝映,如下:

<dubbo:reference id="userService" interface="com.test.UserServiceBo"
        group="dubbo" version="1.0.0" timeout="3000" actives="10"/>

設(shè)置com.test.UserServiceBo接口中所有方法吴叶,每個(gè)方法最多同時(shí)并發(fā)請(qǐng)求10個(gè)請(qǐng)求。

也可以使用下面方法設(shè)置接口中的單個(gè)方法的并發(fā)請(qǐng)求個(gè)數(shù)序臂,如下:


    <dubbo:reference id="userService" interface="com.test.UserServiceBo"
        group="dubbo" version="1.0.0" timeout="3000">
                <dubbo:method name="sayHello" actives="10" />
    </dubbo:reference>

如上設(shè)置sayHello方法的并發(fā)請(qǐng)求數(shù)量最大為10蚌卤,如果客戶端請(qǐng)求該方法并發(fā)超過了10則客戶端會(huì)被阻塞,等客戶端并發(fā)請(qǐng)求數(shù)量少于10的時(shí)候奥秆,該請(qǐng)求才會(huì)被發(fā)送到服務(wù)提供方服務(wù)器逊彭。在dubbo中客戶端并發(fā)控制是使用ActiveLimitFilter過濾器來控制的,代碼如下:

public class ActiveLimitFilter implements Filter {

    public Result invoke(Invoker<?> invoker, Invocation invocation) throws RpcException {
        URL url = invoker.getUrl();
        String methodName = invocation.getMethodName();
        //獲取設(shè)置的acvites的值构订,默認(rèn)為0
        int max = invoker.getUrl().getMethodParameter(methodName, Constants.ACTIVES_KEY, 0);
        //獲取當(dāng)前方法目前并發(fā)請(qǐng)求數(shù)量
        RpcStatus count = RpcStatus.getStatus(invoker.getUrl(), invocation.getMethodName());
        if (max > 0) {//說明設(shè)置了actives變量
            long timeout = invoker.getUrl().getMethodParameter(invocation.getMethodName(), Constants.TIMEOUT_KEY, 0);
            long start = System.currentTimeMillis();
            long remain = timeout;
            int active = count.getActive();
            //如果該方法并發(fā)請(qǐng)求數(shù)量大于設(shè)置值侮叮,則掛起當(dāng)前線程。
            if (active >= max) {
                synchronized (count) {
                    while ((active = count.getActive()) >= max) {
                        try {
                            count.wait(remain);
                        } catch (InterruptedException e) {
                        }
                        //如果等待時(shí)間超時(shí)鲫咽,則拋出異常
                        long elapsed = System.currentTimeMillis() - start;
                        remain = timeout - elapsed;
                        if (remain <= 0) {
                            throw new RpcException("Waiting concurrent invoke timeout in client-side for service:  "
                                    + invoker.getInterface().getName() + ", method: "
                                    + invocation.getMethodName() + ", elapsed: " + elapsed
                                    + ", timeout: " + timeout + ". concurrent invokes: " + active
                                    + ". max concurrent invoke limit: " + max);
                        }
                    }
                }
            }
        }
        //沒有限流時(shí)候签赃,正常調(diào)用
        try {
            long begin = System.currentTimeMillis();
            RpcStatus.beginCount(url, methodName);
            try {
                Result result = invoker.invoke(invocation);
                RpcStatus.endCount(url, methodName, System.currentTimeMillis() - begin, true);
                return result;
            } catch (RuntimeException t) {
                RpcStatus.endCount(url, methodName, System.currentTimeMillis() - begin, false);
                throw t;
            }
        } finally {
            if (max > 0) {
                synchronized (count) {
                    count.notify();
                }
            }
        }
    }

}

可知客戶端并發(fā)控制,是如果當(dāng)并發(fā)量達(dá)到指定值后分尸,當(dāng)前客戶端請(qǐng)求線程會(huì)被掛起锦聊,如果在等待超時(shí)期間并發(fā)請(qǐng)求量少了,那么阻塞的線程會(huì)被激活箩绍,然后發(fā)送請(qǐng)求到服務(wù)提供方孔庭,如果等待超時(shí)了,則直接拋出異常材蛛,這時(shí)候服務(wù)根本都沒有發(fā)送到服務(wù)提供方服務(wù)器圆到。

四、 改進(jìn)的廣播策略

前面我們講解集群容錯(cuò)時(shí)候談到有一個(gè)廣播策略卑吭,該策略主要用于對(duì)所有服務(wù)提供者進(jìn)行廣播消息芽淡,那么有個(gè)問題需要思考,廣播是是說你在客戶端調(diào)用接口一次豆赏,內(nèi)部就是輪詢調(diào)用所有服務(wù)提供者的機(jī)器的服務(wù)挣菲,那么你調(diào)用一次該接口富稻,返回值是什么那?比如內(nèi)部輪詢了10臺(tái)機(jī)器白胀,每個(gè)機(jī)器應(yīng)該都有一個(gè)返回值椭赋,那么你調(diào)用的這一次返回值是10個(gè)返回值的組成?其實(shí)不是或杠,返回的是輪詢調(diào)用的最后一個(gè)機(jī)器結(jié)果哪怔,那么如果我們想把所有的機(jī)器返回的結(jié)果聚合起來如何做的? 使用dubbo中更多需要注意的事情 單擊我查看文章 向抢, 單擊我觀看視頻即可知曉认境。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市笋额,隨后出現(xiàn)的幾起案子元暴,更是在濱河造成了極大的恐慌,老刑警劉巖兄猩,帶你破解...
    沈念sama閱讀 218,546評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件茉盏,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡枢冤,警方通過查閱死者的電腦和手機(jī)鸠姨,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,224評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來淹真,“玉大人讶迁,你說我怎么就攤上這事『苏海” “怎么了巍糯?”我有些...
    開封第一講書人閱讀 164,911評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)客扎。 經(jīng)常有香客問我祟峦,道長(zhǎng),這世上最難降的妖魔是什么徙鱼? 我笑而不...
    開封第一講書人閱讀 58,737評(píng)論 1 294
  • 正文 為了忘掉前任宅楞,我火速辦了婚禮,結(jié)果婚禮上袱吆,老公的妹妹穿的比我還像新娘厌衙。我一直安慰自己,他們只是感情好绞绒,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,753評(píng)論 6 392
  • 文/花漫 我一把揭開白布婶希。 她就那樣靜靜地躺著,像睡著了一般蓬衡。 火紅的嫁衣襯著肌膚如雪喻杈。 梳的紋絲不亂的頭發(fā)上拐揭,一...
    開封第一講書人閱讀 51,598評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音奕塑,去河邊找鬼。 笑死家肯,一個(gè)胖子當(dāng)著我的面吹牛龄砰,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播讨衣,決...
    沈念sama閱讀 40,338評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼换棚,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了反镇?” 一聲冷哼從身側(cè)響起固蚤,我...
    開封第一講書人閱讀 39,249評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎歹茶,沒想到半個(gè)月后夕玩,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,696評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡惊豺,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,888評(píng)論 3 336
  • 正文 我和宋清朗相戀三年燎孟,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片尸昧。...
    茶點(diǎn)故事閱讀 40,013評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡揩页,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出烹俗,到底是詐尸還是另有隱情爆侣,我是刑警寧澤,帶...
    沈念sama閱讀 35,731評(píng)論 5 346
  • 正文 年R本政府宣布幢妄,位于F島的核電站兔仰,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏磁浇。R本人自食惡果不足惜斋陪,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,348評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望置吓。 院中可真熱鬧无虚,春花似錦、人聲如沸衍锚。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,929評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)戴质。三九已至度宦,卻和暖如春踢匣,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背戈抄。 一陣腳步聲響...
    開封第一講書人閱讀 33,048評(píng)論 1 270
  • 我被黑心中介騙來泰國(guó)打工离唬, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人划鸽。 一個(gè)月前我還...
    沈念sama閱讀 48,203評(píng)論 3 370
  • 正文 我出身青樓输莺,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親裸诽。 傳聞我的和親對(duì)象是個(gè)殘疾皇子嫂用,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,960評(píng)論 2 355

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

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)丈冬,斷路器嘱函,智...
    卡卡羅2017閱讀 134,657評(píng)論 18 139
  • 2010年10月5日,我的雷喵喵忽罹惡疾埂蕊,急送寵物醫(yī)院往弓,醫(yī)生查不出病因,終于不忍見他痛苦粒梦,我簽字為他實(shí)施了安樂死...
    葉知秋yzq閱讀 640評(píng)論 1 2
  • 那天你說到覺得我回去要不就是打游戲亮航,要不就是看球,要不就是XXOO匀们,可是小狐你也不跟我溝通缴淋,這樣只會(huì)讓隔閡越來越深...
    maomaoxiong閱讀 357評(píng)論 0 1
  • 1月20日,周六晚泄朴,加一舵主加班輔導(dǎo)進(jìn)行了月度目標(biāo)制定方法的學(xué)習(xí)重抖。感恩~ 第一步:明確劃分月計(jì)劃、周計(jì)劃的時(shí)段 4...
    Sophy_CQ閱讀 893評(píng)論 0 1
  • 伴隨著綿綿細(xì)雨,我從保護(hù)了我二十幾年的“港灣”歸到了人性的世界局扶。這個(gè)世界對(duì)于我來說很是陌生和惶恐恨统,你或許不會(huì)懂我的...
    清湯掛面加煎蛋閱讀 163評(píng)論 0 1