java消息服務(wù)之ActiveMQ入門(0x02)

java消息服務(wù)之ActiveMQ入門(0x01)

概要

  • ActiveMQ 的 request-response通信方式
  • ActiveMQ 的事務(wù)
  • ActiveMQ消息消費的同步和異步
  • ActiveMQ的零散知識點

ActiveMQ 的 request-response通信方式

在前面的兩種模式中都是一方負責(zé)發(fā)送消息而另外一方負責(zé)處理瑟匆。而我們實際中的很多應(yīng)用相當(dāng)于一種一應(yīng)一答的過程所禀,需要雙方都能給對方發(fā)送消息酌心。于是請求-應(yīng)答的這種通信方式也很重要检吆。它也應(yīng)用的很普遍。


首先發(fā)送端代碼

public class ReSend {
    public static void main(String[] args) throws JMSException {
        ActiveMQConnectionFactory connectionFactory = new ActiveMQConnectionFactory("tcp://localhost:61616");  
        Connection connection = connectionFactory.createConnection();  
        connection.start();  
        Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);  
        Destination adminQueue = session.createQueue("shli.1");  
        //創(chuàng)建生產(chǎn)者發(fā)送消息
        MessageProducer producer = session.createProducer(adminQueue);  
        producer.setDeliveryMode(DeliveryMode.NON_PERSISTENT);  
  
        //創(chuàng)建一個臨時的通道蛮瞄,并且創(chuàng)建消費者 
        Destination tempDest = session.createTemporaryQueue();  
        MessageConsumer responseConsumer = session.createConsumer(tempDest);  
  
        //設(shè)置臨時消息的一個監(jiān)聽器  
        responseConsumer.setMessageListener(new MessageListener() {
            @Override
            public void onMessage(Message arg0) {
                try {
                    System.out.println("收到回復(fù)內(nèi)容:"+((TextMessage)arg0).getText());
                } catch (JMSException e) {
                    e.printStackTrace();
                }
                }
            });  
  
        //模擬消息  
        TextMessage txtMessage = session.createTextMessage();  
        txtMessage.setText("我是ReSend");  
  
        //設(shè)置消息頭部信息   使得回復(fù)的消息可以定位到剛剛創(chuàng)建的臨時通道
        txtMessage.setJMSReplyTo(tempDest);  
  
        //這里在消息頭部設(shè)置一個標(biāo)示嚼摩,好像僅僅是用于業(yè)務(wù)上的判斷,并不影響消息的應(yīng)答
        String correlationId = UUID.randomUUID().toString();  
        txtMessage.setJMSCorrelationID(correlationId);  
        producer.send(txtMessage);  
        
    }
}

跟之前P2P不一樣的是發(fā)送端這里有創(chuàng)建一個臨時通道以及臨時通道對應(yīng)的消費者鲫剿,并且在消息的頭部添加了JMSReplyTo和MSCorrelationID,用于接收端進行消息的回復(fù)稻轨。

然后接收端的代碼

public class ReReceive {
    public static void main(String[] args) throws JMSException {
          ActiveMQConnectionFactory connectionFactory = new ActiveMQConnectionFactory("tcp://localhost:61616");  
          Connection connection = connectionFactory.createConnection(); 
          connection.start();  
          final Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);  
          Destination adminQueue = session.createQueue("shli.1");
          
          //創(chuàng)建回復(fù)用的生產(chǎn)者
          final MessageProducer replyProducer = session.createProducer(null);  
          replyProducer.setDeliveryMode(DeliveryMode.NON_PERSISTENT);  
          //創(chuàng)建消費者對消息進行消費 
          MessageConsumer consumer = session.createConsumer(adminQueue);  
          consumer.setMessageListener(new MessageListener() {
            @Override
            public void onMessage(Message message) {
                try {
                    TextMessage response = session.createTextMessage();
                    if (message instanceof TextMessage) {  
                        TextMessage txtMsg = (TextMessage) message;  
                        String messageText = txtMsg.getText(); 
                        System.out.println("收到了來自ReSend的消息------"+ messageText);
                        response.setText("回復(fù)ReSend灵莲,F(xiàn)rom ReReceive");  
                    } 
                     response.setJMSCorrelationID(message.getJMSCorrelationID()); 
                     replyProducer.send(message.getJMSReplyTo(), response);  
                } catch (JMSException e) {
                    e.printStackTrace();
                } 
            }
        });
    }
}

相比以之前的P2P接收端這里多了一個生產(chǎn)者 replyProducer ,然后在收到消息之后殴俱,根據(jù)發(fā)送端發(fā)送過來的消息的頭部信息政冻,返回應(yīng)答信息~

ActiveMQ 的本地事務(wù)

Session,用于發(fā)送和接受消息线欲,而且是單線程的明场,支持事務(wù)的。比如之前的代碼

Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE); 

第一個參數(shù)為true的時候表示開啟事務(wù)李丰,為false的時候表示關(guān)閉事務(wù)苦锨。

在一個JMS客戶端,可以使用本地事務(wù)來組合消息的發(fā)送和接收趴泌。JMS Session接口提供了commit和rollback方法舟舒。事務(wù)提交意味著生產(chǎn)的所有消息被發(fā)送,消費的所有消息被確認(rèn)嗜憔;事務(wù)回滾意味著生產(chǎn)的所有消 息被銷毀秃励,消費的所有消息被恢復(fù)并重新提交,除非它們已經(jīng)過期吉捶。
事務(wù)性的會話總是牽涉到事務(wù)處理中夺鲜,commit或rollback方法一旦被調(diào)用,一個事務(wù)就結(jié)束了呐舔,而另一個事務(wù)被開始币励。關(guān)閉事務(wù)性會話將回滾其中的事務(wù)。
需要注意的是滋早,如果使用請求/回復(fù)機制榄审,即發(fā)送一個消息砌们,同時希望在同一個事務(wù)中等待接收該消息的回復(fù)杆麸,那么程序?qū)⒈粧炱鸶榻驗橹钡绞聞?wù)提交,發(fā)送操作才會真正執(zhí)行昔头。
需要注意的還有一個饼问,消息的生產(chǎn)和消費不能包含在同一個事務(wù)中。

createSession的第二個參數(shù)

在事務(wù)性會話中揭斧,當(dāng)一個事務(wù)被提交的時候莱革,確認(rèn)自動發(fā)生。在非事務(wù)性會話中讹开,消息何時被確認(rèn)取決于創(chuàng)建會話時的應(yīng)答模式(acknowledgement mode)盅视。該參數(shù)有以下三個可選值:
? Session.AUTO_ACKNOWLEDGE。當(dāng)客戶成功的從receive方法返回的時候旦万,或者從MessageListener.onMessage方法成功返回的時候闹击,會話自動確認(rèn)客戶收到的消息。
? Session.CLIENT_ACKNOWLEDGE成艘。 客戶通過消息的acknowledge方法確認(rèn)消息赏半。需要注意的是,在這種模式中淆两,確認(rèn)是在會話層上進行:確認(rèn)一個被消費的消息將自動確認(rèn)所有已被會話消 費的消息断箫。例如,如果一個消息消費者消費了10個消息秋冰,然后確認(rèn)第5個消息仲义,那么所有10個消息都被確認(rèn)。
JMS消息只有在被確認(rèn)之后丹莲,才認(rèn)為已經(jīng)被成功地消費了光坝。消息的成功消費通常包含三個階段:客戶接收消息、客戶處理消息和消息被確認(rèn)甥材。
? Session.DUPS_ACKNOWLEDGE盯另。 該選擇只是會話遲鈍第確認(rèn)消息的提交。如果JMS provider失敗洲赵,那么可能會導(dǎo)致一些重復(fù)的消息鸳惯。如果是重復(fù)的消息,那么JMS provider必須把消息頭的JMSRedelivered字段設(shè)置為true

ActiveMQ消息消費的同步和異步

同步接收消息:接收者主動接收消息叠萍,若消息隊列中沒有消息則阻塞等待芝发,當(dāng)然也有其它方法可以規(guī)定等待時間或是不等待。

  Message message =  messageConsumer1.receive();
這種方式線程會一直阻塞苛谷,直到接收到消息辅鲸,和Socket服務(wù)器端接收客戶端請求的方式差不多

異步接收消息:當(dāng)消息隊列有消息時會調(diào)用接收者的onMessage方法,接收者不用阻塞等待腹殿,可執(zhí)行其它業(yè)務(wù)

 messageConsumer1.setMessageListener(new MessageListener() {
        @Override
        public void onMessage(Message arg0) {
            MapMessage map = (MapMessage)arg0;  
            try {
                String shli = map.getString("name");
                double price = map.getDouble("price");  
    boolean up = map.getBoolean("up");  
    System.out.println(shli + "-1-1-1-" + price +"-1--1-1--"+up+"---Thread---" +Thread.currentThread().getName());  
            } catch (JMSException e) {
                e.printStackTrace();
            }  
            
        }
    });
這種方式不會出現(xiàn)線程阻塞独悴,當(dāng)消息到達時例书,會在其他線程中執(zhí)行onMessage()中的邏輯

ActiveMQ的零散功能點

消息過期 和 優(yōu)先級

可以設(shè)置消息在一定時間后過期,默認(rèn)是永不過期刻炒。設(shè)置方法 后兩個send方法的最后一個參數(shù)

可以使用消息優(yōu)先級來指示JMS provider首先提交緊急的消息决采。優(yōu)先級分10個級別,從0(最低)到9(最高)坟奥。如果不指定優(yōu)先級树瞭,默認(rèn)級別是4。需要注意的是爱谁,JMS provider并不一定保證按照優(yōu)先級的順序提交消息晒喷。設(shè)置方法 后兩個send方法的最后第三個參數(shù)

JMS Selectors

消息對象有消息屬性,用于消息選擇器

使用如下:

  1. 生產(chǎn)者端需要設(shè)置消息屬性访敌,一定要注意的是setXxxProperty(filed,value)

    message.setIntProperty("pa", 6);

  2. 給出條件厨埋,其實本質(zhì)上就是SQL92語法

    String condition = "pa > 7";

  3. 創(chuàng)建消費者的時候,指定條件即可

    MessageConsumer messageConsumer1 = session.createConsumer(destination,condition);

持久性

? PERSISTENT捐顷。指示JMS provider持久保存消息荡陷,以保證消息不會因為JMS provider的失敗而丟失。
? NON_PERSISTENT迅涮。不要求JMS provider持久保存消息

Producer.setDeliveryMode(DeliveryMode.NON_PERSISTENT);

持久化訂閱

在第一篇的代碼中我們發(fā)現(xiàn)废赞,在訂閱模式中,如果消費者不在線叮姑,在上線之后是沒有辦法重新收到生產(chǎn)者發(fā)送的消息的唉地。那是因為我們沒有將消費者設(shè)置為持久化訂閱。

客戶可以通過會話上的createDurableSubscriber方法來創(chuàng)建一個持久訂閱传透,該方法的第一個參數(shù)必須是一個topic耘沼。第二個參數(shù)是訂閱的名稱。



這樣設(shè)置之后朱盐,消費者上線后就可以收到之前生產(chǎn)者發(fā)送的(離線)消息群嗤。
但是這種沒有持久性的消息只是被緩存在內(nèi)存中,如果ActiveMQ被關(guān)閉并重新打開的話兵琳,消費者同樣接收不到(離線)消息狂秘。

如果在發(fā)送端加上一句代碼,這樣就算是ActiveMQ重啟也不會出現(xiàn)消息丟失的現(xiàn)象了

Producer.setDeliveryMode(DeliveryMode.PERSISTENT);

JMS provider會存儲發(fā)布到持久訂閱對應(yīng)的topic上的消息躯肌。如果最初創(chuàng)建持久訂閱的客戶或者任何其它客戶使用相同的連接工廠和連接的客戶ID者春、相同 的主題和相同的訂閱名再次調(diào)用會話上的createDurableSubscriber方法,那么該持久訂閱就會被激活清女。JMS provider會象客戶發(fā)送客戶處于非激活狀態(tài)時所發(fā)布的消息钱烟。
持久訂閱在某個時刻只能有一個激活的訂閱者。持久訂閱在創(chuàng)建之后會一直保留,直到應(yīng)用程序調(diào)用會話上的unsubscribe方法拴袭。

最后

下一遍將記錄ActiveMQ和其他框架的整合

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末传惠,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子稻扬,更是在濱河造成了極大的恐慌,老刑警劉巖羊瘩,帶你破解...
    沈念sama閱讀 217,509評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件泰佳,死亡現(xiàn)場離奇詭異,居然都是意外死亡尘吗,警方通過查閱死者的電腦和手機逝她,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評論 3 394
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來睬捶,“玉大人黔宛,你說我怎么就攤上這事∏苊常” “怎么了臀晃?”我有些...
    開封第一講書人閱讀 163,875評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長介劫。 經(jīng)常有香客問我徽惋,道長,這世上最難降的妖魔是什么座韵? 我笑而不...
    開封第一講書人閱讀 58,441評論 1 293
  • 正文 為了忘掉前任险绘,我火速辦了婚禮,結(jié)果婚禮上誉碴,老公的妹妹穿的比我還像新娘宦棺。我一直安慰自己,他們只是感情好黔帕,可當(dāng)我...
    茶點故事閱讀 67,488評論 6 392
  • 文/花漫 我一把揭開白布代咸。 她就那樣靜靜地躺著,像睡著了一般成黄。 火紅的嫁衣襯著肌膚如雪侣背。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,365評論 1 302
  • 那天慨默,我揣著相機與錄音贩耐,去河邊找鬼。 笑死厦取,一個胖子當(dāng)著我的面吹牛潮太,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 40,190評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼铡买,長吁一口氣:“原來是場噩夢啊……” “哼更鲁!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起奇钞,我...
    開封第一講書人閱讀 39,062評論 0 276
  • 序言:老撾萬榮一對情侶失蹤澡为,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后景埃,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體媒至,經(jīng)...
    沈念sama閱讀 45,500評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,706評論 3 335
  • 正文 我和宋清朗相戀三年谷徙,在試婚紗的時候發(fā)現(xiàn)自己被綠了拒啰。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,834評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡完慧,死狀恐怖谋旦,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情屈尼,我是刑警寧澤册着,帶...
    沈念sama閱讀 35,559評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站脾歧,受9級特大地震影響指蚜,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜涨椒,卻給世界環(huán)境...
    茶點故事閱讀 41,167評論 3 328
  • 文/蒙蒙 一摊鸡、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蚕冬,春花似錦免猾、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,779評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至旁蔼,卻和暖如春锨苏,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背棺聊。 一陣腳步聲響...
    開封第一講書人閱讀 32,912評論 1 269
  • 我被黑心中介騙來泰國打工伞租, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人限佩。 一個月前我還...
    沈念sama閱讀 47,958評論 2 370
  • 正文 我出身青樓葵诈,卻偏偏與公主長得像裸弦,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子作喘,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,779評論 2 354

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