只对比较简单的direct模式进行了封装,队列支持持久化、镜像以及auto-delete。
(但是不支持交换机的auto-delete,不知道扩展中为什么没与支持)
先执行消费者,创建对交换机、队列并与路由键绑定,并监听消息队列;
再执行生产者,向已创建的交换机中发送消息。
直接上代码了:
/**
* 消费者类
*/
class AsynMessageConsumer{
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;
/**
* 创建连接、交换机、队列,并绑定
* @param array $config RabbitMQ服务器信息
* @param string $e_name 交换机名称
* @param string $k_route 路由键
* @param string $q_name 队列名称
* @param bool $durable 队列是否持久化
* @param bool $mirror 队列是否镜像
* @return void
*/
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;
$this->config = $config;
if (!self::connect()) return False;
$this->channel = new AMQPChannel($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;
}
/**
* 循环阻塞方式接收消息
* @param string $fun_name 自定义处理函数的函数名
* @param bool $autoack 是否自动发送ACK应答,否则需要在自定义处理函数中手动发送
* @return bool
*/
public function run($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秒等待)
* @param int $i 配置项索引
* @return bool
*/
private function connect($i = 0){
if (array_key_exists($i, $this->config)){
try{
$this->conn = new AMQPConnection($this->config[$i]);
$this->conn->connect();
$ret = True;
}catch(AMQPConnectionException $e){
$ret = $this->connect(++$i);
}
} else {
$ret = False;
}
return $ret;
}
/**
* 创建交换机
* @param string $name 名称
* @return int
*/
private function establishExchange($name){
$ex = new AMQPExchange($this->channel);
$ex->setName($name);
$ex->setType(AMQP_EX_TYPE_DIRECT); //direct类型
if ($this->durable) $ex->setFlags(AMQP_DURABLE); //持久化
return $ex->declare();
}
/**
* 创建队列
* @param string $name 名称
* @param string $e_name 交换机名称
* @param string $k_route 路由键
* @return int
*/
private function establishQueue($name, $e_name, $k_route){
$this->queue = new AMQPQueue($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->declare();
$ret = $this->queue->bind($e_name, $k_route);
return $ret;
}
public function __destruct(){
if ($this->conn){
$this->conn->disconnect();
}
}
}
/**
* 生产者类
*/
class AsynMessagePublisher{
private $config = array();
private $conn = Null;
private $channel = Null;
private $exchange = Null;
public $is_ready = False;
/**
* 创建连接,并指定交换机
* @param array $config RabbitMQ服务器信息
* @param string $e_name 交换机名称
* @return void
*/
public function __construct($config, $e_name){
if (!($config && $e_name)) return False;
$this->config = $config;
if (!self::connect()) return False;
$this->channel = new AMQPChannel($this->conn);
$this->establishExchange($e_name);
$this->is_ready = True;
}
/**
* 发送消息
* @param string $msg 消息体
* @param string $k_route 路由键
* @return int / False
*/
public function send($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秒等待)
* @param int $i 配置项索引
* @return bool
*/
private function connect($i = 0){
if (array_key_exists($i, $this->config)){
try{
$this->conn = new AMQPConnection($this->config[$i]);
$this->conn->connect();
$ret = True;
}catch(AMQPConnectionException $e){
$ret = $this->connect(++$i);
}
} else {
$ret = False;
}
return $ret;
}
/**
* 创建交换机
* @param string $name 名称
* @return void
*/
private function establishExchange($name){
$this->exchange = new AMQPExchange($this->channel);
$this->exchange->setName($name);
}
public function __destruct(){
if ($this->conn){
$this->conn->disconnect();
}
}
}
使用比较简单,demo例子就不给了。
实际应用中也可以再用工厂模式进行改造。