4、简单封装RabbitMQ的发布与订阅模式

生产者类:Publisher.class.php

classPublisher{

          private $config=array();

          private $conn=Null;

          private $channel=Null;

          private $exchange=Null;

          public $is_ready=False;

         /**

          * 创建连接,并指定交换机

          * @paramarray $config RabbitMQ服务器信息

          * @paramstring $e_name交换机名称

          * @returnvoid

          */

          public function__construct($config,$e_name){

                  if(!($config&&$e_name)) {

                           return False;

                  }

                  shuffle($config);

                  $this->config=$config;

                  if(!self::connect()) {

                          return False;

                  }

                  $this->channel=newAMQPChannel($this->conn);

                  $this->establishExchange($e_name);

                  $this->is_ready=True;

          }

          /**

           * 发送消息

           * @paramstring $msg消息体

           * @paramstring $k_route路由键

           * @returnint / False

           */

           public functionsend($msg,$k_route){

                   $msg=trim(strval($msg));

                   if(!$this->exchange||$msg===''|| !$k_route) return False;

                   $ret=$this->exchange->publish($msg,$k_route);

                   return $ret;

           }

           /**

            * 创建链接

            * 无法链接时则会自动选择下一个配置项(IP不通的情况下会有5秒等待)

            * @paramint $i配置项索引

            * @returnbool

            */

            private functionconnect($i=0){

                    if(array_key_exists($i,$this->config)){

                             try{

                                            $this->conn=newAMQPConnection($this->config[$i]);

                                            $this->conn->connect();

                                            $ret=True;

                              }catch(AMQPConnectionException$e){

                                            $ret=$this->connect(++$i);

                              }

                     }else{

                             $ret=False;

                     }

                     return$ret;

            }

            /**

             * 创建交换机

             * @paramstring $name名称

             * @returnvoid

             */

             private functionestablishExchange($name){

                       $this->exchange=newAMQPExchange($this->channel);

                       $this->exchange->setName($name);

              }

              public function__destruct(){

                       if($this->conn){

                                    $this->conn->disconnect();

                        }

               }

}

消费者类:Consumer.class.php

class Consumer {

           private$config=array();

           private$durable=True;

           private$mirror=False;

           private$autodelete=False;

           private$conn=Null;

           private$channel=Null;

           private$queue=Null;

           public$is_ready=False;

           /**

            * 创建连接、交换机、队列,并绑定

            * @paramarray $config RabbitMQ服务器信息

            * @paramstring $e_name交换机名称

            * @paramstring $k_route路由键

            * @paramstring $q_name队列名称

            * @parambool $durable队列是否持久化

            * @parambool $mirror队列是否镜像

            * @returnvoid

            */

            public function__construct($config,$e_name,$k_route,$q_name,$durable=True,$mirror=False,$autodelete=False){

                         if(!($config&&$e_name&&$q_name&&$k_route)){

                                       return False;

                          }

                          shuffle($config);

                          $this->config=$config;

                          if(!self::connect()){

                                      return False;

                          }

                         $this->channel=newAMQPChannel($this->conn);

                         $this->durable= (bool)$durable;

                         $this->mirror= (bool)$mirror;

                         $this->autodelete= (bool)$autodelete;

                        $this->establishExchange($e_name);

                        $this->establishQueue($q_name,$e_name,$k_route);

                        $this->is_ready=True;

             }

            /**

             * 循环阻塞方式接收消息

             * @paramstring $fun_name自定义处理函数的函数名

              * @parambool $autoack是否自动发送ACK应答,否则需要在自定义处理函数中手动发送

              * @returnbool

              */

              public functionrun($fun_name,$autoack=True){

                         $fun_name=strval($fun_name);

                         if(!$fun_name|| !$this->queue)return False;

                         while(True){

                                    if($autoack)$this->queue->consume($fun_name,AMQP_AUTOACK);

                                    else$this->queue->consume($fun_name);

                         }

                }

                /**

                 * 创建链接

                 * 无法链接时则会自动选择下一个配置项(IP不通的情况下会有5秒等待)

                 * @paramint $i配置项索引

                 * @returnbool

                 */

                 private functionconnect($i=0){

                            if(array_key_exists($i,$this->config)){

                                     try{

                                                   $this->conn=newAMQPConnection($this->config[$i]);

                                                   $this->conn->connect();

                                                   $ret=True;

                                       }catch(AMQPConnectionException$e){

                                                    $ret=$this->connect(++$i);

                                       }

                           }else{

                                       $ret=False;

                           }

                          return $ret;

                }

               /**

                * 创建交换机

                * @paramstring $name名称

                * @returnint

                */

                private functionestablishExchange($name){

                           $ex=newAMQPExchange($this->channel);

                           $ex->setName($name);

                           $ex->setType(AMQP_EX_TYPE_DIRECT);//direct类型

                           if($this->durable) $ex->setFlags(AMQP_DURABLE);//持久化

                           //return $ex->declareExchange();

                           return true;

                }

                /**

                 * 创建队列

                 * @paramstring $name名称

                 * @paramstring $e_name交换机名称

                 * @paramstring $k_route路由键

                 * @returnint

                 */

                 private functionestablishQueue($name,$e_name,$k_route){

                             $this->queue=newAMQPQueue($this->channel);

                             $this->queue->setName($name);

                             if($this->durable)$this->queue->setFlags(AMQP_DURABLE);//持久化

                             if($this->mirror)$this->queue->setArgument('x-ha-policy','all');//镜像

                              if($this->autodelete)$this->queue->setFlags(AMQP_AUTODELETE);//auto-delete

                              $this->queue->declareQueue();

                              $ret=$this->queue->bind($e_name,$k_route);

                              return$ret;

                  }

                  public function__destruct(){

                            if($this->conn){

                                           $this->conn->disconnect();

                             }

                   }

}

demoC.php

include_once'./Consumer.class.php';

functionlogResult($word='') {

           $fp=fopen("log.txt","a");

           flock($fp,LOCK_EX) ;

           fwrite($fp,"执行日期:".strftime("%Y-%m-%d %H:%M:%S",time())."\n".$word."\n");

           flock($fp,LOCK_UN);

           fclose($fp);

}

$config=array(

         array(

                         'host'=>'127.0.0.1',

                         'port'=>'5672',

                         'login'=>'ybl',

                         'password'=>'ybl',

                         'vhost'=>'/'

         ),

         array(

                         'host'=>'127.0.0.2',

                         'port'=>'5672',

                         'login'=>'ybl',

                         'password'=>'ybl',

                         'vhost'=>'/'

          )

);

$e_name='demo';//交换机名

$q_name='ybl';//队列名

$k_route='hello';//路由key

if(!$cs=newConsumer($config,$e_name,$k_route,$q_name)){

             exit("error");

}

//第二个参数默认为true,自动发送ACK应答

$cs->run('dealMessage');

//消费回调函数,处理消息

functiondealMessage($envelope,$queue) {

$msg=$envelope->getBody();

//记录log日记

logResult($msg);

$queue->ack($envelope->getDeliveryTag());//手动发送ACK应答

}

demoP.php

include_once'./Publisher.class.php';

$config=array(

       array(

               'host'=>'127.0.0.1',

               'port'=>'5672',

               'login'=>'ybl',

               'password'=>'ybl',

               'vhost'=>'/'

      ),

      array(

              'host'=>'127.0.0.2',

              'port'=>'5672',

              'login'=>'ybl',

              'password'=>'ybl',

              'vhost'=>'/'

         )

);

$e_name='demo';//交换机名

$k_route='hello';//路由key

if(!$conn=newPublisher($config,$e_name)){

           echo'error';

           exit;

}

$msg='hello RabbitMQ';

for($i=0;$i<10;$i++){

           $res=$conn->send($msg,$k_route);

           ob_flush();

           flush();

           echo $res;

           sleep(1);

}

运行脚本demoP.php   demoC.php查看

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,117评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,328评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,839评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,007评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,384评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,629评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,880评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,593评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,313评论 1 243
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,575评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,066评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,392评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,052评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,082评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,844评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,662评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,575评论 2 270

推荐阅读更多精彩内容