YII使用beanstalk队列

本文介绍了如何在CentOS系统上通过YUM安装beanstalk,并在YII框架中集成beanstalk队列。首先,通过yum命令安装beanstalk,然后下载并配置beanstalk的PHP扩展,将其放置在extensions目录下。虽然无法提供附件,但提供了PHP版本的队列核心代码。

1.系统centos 我是直接使用yum install beanstalk安装的

2.下载beanstalk的php扩展包 放在extensions


//控制器往队列里面塞任务  @author yao
Yii::import('application.extensions.SendSmsBeanstalk');

class AController extends CController{

    public function actionIndex(){
        for($i=0;$i<1000;$i++){
            SendSmsBeanstalk::sendSms('sendsms_'.$i);
        }
    }

    public function actionA(){
        SendSmsBeanstalk::handleMessage();
    }
}
</pre><pre code_snippet_id="546591" snippet_file_name="blog_20141209_4_7698245" name="code" class="php">//发送短信  @author yao
Yii::import('application.extensions.beanstalk.*');
class SendSmsBeanstalk extends CComponent{

    //建立发短信任务
    public static function sendSms($body){
        //实例化beanstalk
        $beanstalk = new Socket_Beanstalk();
        if (!$beanstalk->connect()) {
            exit(current($beanstalk->errors()));
        }
        //选择使用的tube
        $beanstalk->useTube('hube_send_smss');


        //往tube中增加数据
        $put = $beanstalk->put(
            23, // 任务的优先级.
            0, // 不等待直接放到ready队列中.
            60, // 处理任务的时间.
            $body // 任务内容
        );
        if (!$put) {
            return false; 
        }
        $beanstalk->disconnect();
    }


    //处理消息
    public static function handleMessage(){
        //实例化beanstalk
        $beanstalk = new Socket_Beanstalk();

        if (!$beanstalk->connect()) {
            exit(current($beanstalk->errors()));
        }

        $beanstalk->useTube('hube_send_smss');
        //设置要监听的tube
        $beanstalk->watch('hube_send_smss');
        //取消对默认tube的监听,可以省略
        $beanstalk->ignore('default');



        while($job = $beanstalk->reserve(2)){//这里我写了单个任务只执行2秒。防止超时。本地测试的时候 没写超时会一直执行下去,哪怕队列里面已经没有任务
            //处理任务
            $result = $job['body'];
            echo $job['id'];echo '<br>';//打印任务ID
            if ($result) {<span style="font-family: Arial, Helvetica, sans-serif;">//这里可以写逻辑 todo</span>
            //删除任务
                $beanstalk->delete($job['id']);
            } else {
            //休眠任务
                $beanstalk->bury($job['id'],'');
            }

        }
        $beanstalk->disconnect();
    }


下方提供我找到的队列PHP版本core,不能上传附件。代码在下面


/**
 * beanstalk: A minimalistic PHP beanstalk client.
 *
 * Copyright (c) 2009-2012 David Persson
 *
 * Distributed under the terms of the MIT License.
 * Redistributions of files must retain the above copyright notice.
 *
 * @copyright  2009-2012 David Persson <nperson@gmx.de>
 * @license    http://www.opensource.org/licenses/mit-license.php The MIT License
 * @link       http://github.com/davidpersson/beanstalk
 */

/**
 * An interface to the beanstalk queue service. Implements the beanstalk
 * protocol spec 1.2. Where appropriate the documentation from the protcol has
 * been added to the docblocks in this class.
 *
 * @link https://github.com/kr/beanstalkd/blob/master/doc/protocol.txt
 */
class Socket_Beanstalk {

    /**
     * Holds a boolean indicating whether a connection to the server is
     * currently established or not.
     *
     * @var boolean
     */
    public $connected = false;

    /**
     * Holds configuration values.
     *
     * @var array
     */
    protected $_config = array();

    /**
     * The current connection resource handle (if any).
     *
     * @var resource
     */
    protected $_connection;

    /**
     * Generated errors. Will hold a maximum of 200 error messages at any time
     * to prevent pilling up messages and using more and more memory. This is
     * especially important if this class is used in long-running workers.
     *
     * @see Socket_Beanstalk::errors()
     * @see Socket_Beanstalk::_error()
     * @var array
     */
    protected $_errors = array();

    /**
     * Constructor.
     *
     * @param array $config An array of configuration values:
     *        - `'persistent'`  Whether to make the connection persistent or
     *                          not, defaults to `true` as the FAQ recommends
     *                          persistent connections.
     *        - `'host'`        The beanstalk server hostname or IP address to
     *                          connect to, defaults to `127.0.0.1`.
     *        - `'port'`        The port of the server to connect to, defaults
     *                          to `11300`.
     *        - `'timeout'`     Timeout in seconds when establishing the
     *                          connection, defaults to `1`.
     * @return void
     */
    public function __construct(array $config = array()) {
        $defaults = array(
            'persistent' => true,
            'host' => '127.0.0.1',
            'port' => 11300,
            'timeout' => 1
        );
        $this->_config = $config + $defaults;
    }

    /**
     * Destructor, disconnects from the server.
     *
     * @return void
     */
    public function __destruct() {
        $this->disconnect();
    }

    /**
     * Initiates a socket connection to the beanstalk server. The resulting
     * stream will not have any timeout set on it. Which means it can wait an
     * unlimited amount of time until a packet becomes available. This is
     * required for doing blocking reads.
     *
     * @see Socket_Beanstalk::$_connection
     * @see Socket_Beanstalk::reserve()
     * @return boolean `true` if the connection was established, `false` otherwise.
     */
    public function connect() {
        if (isset($this->_connection)) {
            $this->disconnect();
        }

        $function = $this->_config['persistent'] ? 'pfsockopen' : 'fsockopen';
        $params = array($this->_config['host'], $this->_config['port'], &$errNum, &$errStr);

        if ($this->_config['timeout']) {
            $params[] = $this->_config['timeout'];
        }
        $this->_connection = @call_user_func_array($function, $params);

        if (!empty($errNum) || !empty($errStr)) {
            $this->_error("{$errNum}: {$errStr}");
        }

        $this->connected = is_resource($this->_connection);

        if ($this->connected) {
            stream_set_timeout($this->_connection, -1);
        }
        return $this->connected;
    }

    /**
     * Closes the connection to the beanstalk server.
     *
     * @return boolean `true` if diconnecting was successful.
     */
    public function disconnect() {
        if (!is_resource($this->_connection)) {
            $this->connected = false;
        } else {
            $this->connected = !fclose($this->_connection);

            if (!$this->connected) {
                $this->_connection = null;
            }
        }
        return !$this->connected;
    }

    /**
     * Returns collected error messages.
     *
     * @return array An array of error messages.
     */
    public function errors() {
        return $this->_errors;
    }

    /**
     * Pushes an error message to `Beanstalk::$_errors`. Ensures
     * that at any point there are not more than 200 messages.
     *
     * @param string $message The error message.
     * @return void
     */
    protected function _error($message) {
        if (count($this->_errors) >= 200) {
            array_shift($this->_errors);
        }
        array_push($this->_errors, $message);
    }

    /**
     * Writes a packet to the socket. Prior to writing to the socket will check
     * for availability of the connection.
     *
     * @param string $data
     * @return integer|boolean number of written bytes or `false` on error.
     */
    protected function _write($data) {
        if (!$this->connected && !$this->connect()) {
            return false;
        }

        $data .= "\r\n";
        return fwrite($this->_connection, $data, strlen($data));
    }

    /**
     * Reads a packet from the socket. Prior to reading from the socket will
     * check for availability of the connection.
     *
     * @param int $length Number of bytes to read.
     * @return string|boolean Data or `false` on error.
     */
    protected function _read($length = null) {
        if (!$this->connected && !$this->connect()) {
            return false;
        }
        if ($length) {
            if (feof($this->_connection)) {
                return false;
            }
            $data = fread($this->_connection, $length + 2);
            $meta = stream_get_meta_data($this->_connection);

            if ($meta['timed_out']) {
                $this->_error('Connection timed out.');
                return false;
            }
            $packet = rtrim($data, "\r\n");
        } else {
            $packet = stream_get_line($this->_connection, 16384, "\r\n");
        }
        return $packet;
    }

    /* Producer Commands */

    /**
     * The `put` command is for any process that wants to insert a job into the queue.
     *
     * @param integer $pri Jobs with smaller priority values will be scheduled
     *        before jobs with larger priorities. The most urgent priority is
     *        0; the least urgent priority is 4294967295.
     * @param integer $delay Seconds to wait before putting the job in the
     *        ready queue.  The job will be in the "delayed" state during this time.
     * @param integer $ttr Time to run - Number of seconds to allow a worker to
     *        run this job.  The minimum ttr is 1.
     * @param string $data The job body.
     * @return integer|boolean `false` on error otherwise an integer indicating
     *         the job id.
     */
    public function put($pri, $delay, $ttr, $data) {
        $this->_write(sprintf('put %d %d %d %d', $pri, $delay, $ttr, strlen($data)));
        $this->_write($data);
        $status = strtok($this->_read(), ' ');

        switch ($status) {
            case 'INSERTED':
            case 'BURIED':
                return (integer)strtok(' '); // job id
            case 'EXPECTED_CRLF':
            case 'JOB_TOO_BIG':
            default:
                $this->_error($status);
                return false;
        }
    }

    /**
     * The `use` command is for producers. Subsequent put commands will put jobs into
     * the tube specified by this command. If no use command has been issued, jobs
     * will be put into the tube named `default`.
     *
     * Please note that while obviously this method should better be named
     * `use` it is not. This is because `use` is a reserved keyword in PHP.
     *
     * @param string $tube A name at most 200 bytes. It specifies the tube to
     *        use.  If the tube does not exist, it will be created.
     * @return string|boolean `false` on error otherwise the name of the tube.
     */
    public function choose($tube) {
        $this->_write(sprintf('use %s', $tube));
        $status = strtok($this->_read(), ' ');

        switch ($status) {
            case 'USING':
                return strtok(' ');
            default:
                $this->_error($status);
                return false;
        }
    }

    /**
     * Alias for choose.
     *
     * @see Socket_Beanstalk::choose()
     * @param string $tube
     * @return string|boolean
     */
    public function useTube($tube) {
        return $this->choose($tube);
    }

    /* Worker Commands */

    /**
     * Reserve a job (with a timeout)
     *
     * @param integer $timeout If given specifies number of seconds to wait for
     *        a job. 0 returns immediately.
     * @return array|false `false` on error otherwise an array holding job id
     *         and body.
     */
    public function reserve($timeout = null) {
        if (isset($timeout)) {
            $this->_write(sprintf('reserve-with-timeout %d', $timeout));
        } else {
            $this->_write('reserve');
        }
        $status = strtok($this->_read(), ' ');

        switch ($status) {
            case 'RESERVED':
                return array(
                    'id' => (integer)strtok(' '),
                    'body' => $this->_read((integer)strtok(' '))
                );
            case 'DEADLINE_SOON':
            case 'TIMED_OUT':
            default:
                $this->_error($status);
                return false;
        }
    }

    /**
     * Removes a job from the server entirely.
     *
     * @param integer $id The id of the job.
     * @return boolean `false` on error, `true` on success.
     */
    public function delete($id) {
        $this->_write(sprintf('delete %d', $id));
        $status = $this->_read();

        switch ($status) {
            case 'DELETED':
                return true;
            case 'NOT_FOUND':
            default:
                $this->_error($status);
                return false;
        }
    }

    /**
     * Puts a reserved job back into the ready queue.
     *
     * @param integer $id The id of the job.
     * @param integer $pri Priority to assign to the job.
     * @param integer $delay Number of seconds to wait before putting the job in the ready queue.
     * @return boolean `false` on error, `true` on success.
     */
    public function release($id, $pri, $delay) {
        $this->_write(sprintf('release %d %d %d', $id, $pri, $delay));
        $status = $this->_read();

        switch ($status) {
            case 'RELEASED':
            case 'BURIED':
                return true;
            case 'NOT_FOUND':
            default:
                $this->_error($status);
                return false;
        }
    }

    /**
     * Puts a job into the `buried` state Buried jobs are put into a FIFO
     * linked list and will not be touched until a client kicks them.
     *
     * @param integer $id The id of the job.
     * @param integer $pri *New* priority to assign to the job.
     * @return boolean `false` on error, `true` on success.
     */
    public function bury($id, $pri) {
        $this->_write(sprintf('bury %d %d', $id, $pri));
        $status = $this->_read();

        switch ($status) {
            case 'BURIED':
                return true;
            case 'NOT_FOUND':
            default:
                $this->_error($status);
                return false;
        }
    }

    /**
     * Allows a worker to request more time to work on a job
     *
     * @param integer $id The id of the job.
     * @return boolean `false` on error, `true` on success.
     */
    public function touch($id) {
        $this->_write(sprintf('touch %d', $id));
        $status = $this->_read();

        switch ($status) {
            case 'TOUCHED':
                return true;
            case 'NOT_TOUCHED':
            default:
                $this->_error($status);
                return false;
        }
    }

    /**
     * Adds the named tube to the watch list for the current
     * connection.
     *
     * @param string $tube Name of tube to watch.
     * @return integer|boolean `false` on error otherwise number of tubes in watch list.
     */
    public function watch($tube) {
        $this->_write(sprintf('watch %s', $tube));
        $status = strtok($this->_read(), ' ');

        switch ($status) {
            case 'WATCHING':
                return (integer)strtok(' ');
            default:
                $this->_error($status);
                return false;
        }
    }

    /**
     * Remove the named tube from the watch list.
     *
     * @param string $tube Name of tube to ignore.
     * @return integer|boolean `false` on error otherwise number of tubes in watch list.
     */
    public function ignore($tube) {
        $this->_write(sprintf('ignore %s', $tube));
        $status = strtok($this->_read(), ' ');

        switch ($status) {
            case 'WATCHING':
                return (integer)strtok(' ');
            case 'NOT_IGNORED':
            default:
                $this->_error($status);
                return false;
        }
    }

    /* Other Commands */

    /**
     * Inspect a job by its id.
     *
     * @param integer $id The id of the job.
     * @return string|boolean `false` on error otherwise the body of the job.
     */
    public function peek($id) {
        $this->_write(sprintf('peek %d', $id));
        return $this->_peekRead();
    }

    /**
     * Inspect the next ready job.
     *
     * @return string|boolean `false` on error otherwise the body of the job.
     */
    public function peekReady() {
        $this->_write('peek-ready');
        return $this->_peekRead();
    }

    /**
     * Inspect the job with the shortest delay left.
     *
     * @return string|boolean `false` on error otherwise the body of the job.
     */
    public function peekDelayed() {
        $this->_write('peek-delayed');
        return $this->_peekRead();
    }

    /**
     * Inspect the next job in the list of buried jobs.
     *
     * @return string|boolean `false` on error otherwise the body of the job.
     */
    public function peekBuried() {
        $this->_write('peek-buried');
        return $this->_peekRead();
    }

    /**
     * Handles response for all peek methods.
     *
     * @return string|boolean `false` on error otherwise the body of the job.
     */
    protected function _peekRead() {
        $status = strtok($this->_read(), ' ');

        switch ($status) {
            case 'FOUND':
                return array(
                    'id' => (integer)strtok(' '),
                    'body' => $this->_read((integer)strtok(' '))
                );
            case 'NOT_FOUND':
            default:
                $this->_error($status);
                return false;
        }
    }

    /**
     * Moves jobs into the ready queue (applies to the current tube).
     *
     * If there are buried jobs those get kicked only otherwise
     * delayed jobs get kicked.
     *
     * @param integer $bound Upper bound on the number of jobs to kick.
     * @return integer|boolean False on error otherwise number of job kicked.
     */
    public function kick($bound) {
        $this->_write(sprintf('kick %d', $bound));
        $status = strtok($this->_read(), ' ');

        switch ($status) {
            case 'KICKED':
                return (integer)strtok(' ');
            default:
                $this->_error($status);
                return false;
        }
    }

    /* Stats Commands */

    /**
     * Gives statistical information about the specified job if it exists.
     *
     * @param integer $id The job id
     * @return string|boolean `false` on error otherwise a string with a yaml formatted dictionary
     */
    public function statsJob($id) {
        $this->_write(sprintf('stats-job %d', $id));
        return $this->_statsRead();
    }

    /**
     * Gives statistical information about the specified tube if it exists.
     *
     * @param string $tube Name of the tube.
     * @return string|boolean `false` on error otherwise a string with a yaml formatted dictionary.
     */
    public function statsTube($tube) {
        $this->_write(sprintf('stats-tube %s', $tube));
        return $this->_statsRead();
    }

    /**
     * Gives statistical information about the system as a whole.
     *
     * @return string|boolean `false` on error otherwise a string with a yaml formatted dictionary.
     */
    public function stats() {
        $this->_write('stats');
        return $this->_statsRead();
    }

    /**
     * Returns a list of all existing tubes.
     *
     * @return string|boolean `false` on error otherwise a string with a yaml formatted list.
     */
    public function listTubes() {
        $this->_write('list-tubes');
        return $this->_statsRead();
    }

    /**
     * Returns the tube currently being used by the producer.
     *
     * @return string|boolean `false` on error otherwise a string with the name of the tube.
     */
    public function listTubeUsed() {
        $this->_write('list-tube-used');
        $status = strtok($this->_read(), ' ');

        switch ($status) {
            case 'USING':
                return strtok(' ');
            default:
                $this->_error($status);
                return false;
        }
    }

    /**
     * Alias for listTubeUsed.
     *
     * @see Socket_Beanstalk::listTubeUsed()
     * @return string|boolean `false` on error otherwise a string with the name of the tube.
     */
    public function listTubeChosen() {
        return $this->listTubeUsed();
    }

    /**
     * Returns a list of tubes currently being watched by the worker.
     *
     * @return string|boolean `false` on error otherwise a string with a yaml formatted list.
     */
    public function listTubesWatched() {
        $this->_write('list-tubes-watched');
        return $this->_statsRead();
    }

    /**
     * Handles responses for all stat methods.
     *
     * @param boolean $decode Whether to decode data before returning it or not. Default is `true`.
     * @return array|string|boolean `false` on error otherwise statistical data.
     */
    protected function _statsRead($decode = true) {
        $status = strtok($this->_read(), ' ');

        switch ($status) {
            case 'OK':
                $data = $this->_read((integer)strtok(' '));
                return $decode ? $this->_decode($data) : $data;
            default:
                $this->_error($status);
                return false;
        }
    }

    /**
     * Decodes YAML data. This is a super naive decoder which just works on a
     * subset of YAML which is commonly returned by beanstalk.
     *
     * @param string $data The data in YAML format, can be either a list or a dictionary.
     * @return array An (associative) array of the converted data.
     */
    protected function _decode($data) {
        $data = array_slice(explode("\n", $data), 1);
        $result = array();

        foreach ($data as $key => $value) {
            if ($value[0] === '-') {
                $value = ltrim($value, '- ');
            } elseif (strpos($value, ':') !== false) {
                list($key, $value) = explode(':', $value);
                $value = ltrim($value, ' ');
            }
            if (is_numeric($value)) {
                $value = (integer) $value == $value ? (integer) $value : (float) $value;
            }
            $result[$key] = $value;
        }
        return $result;
    }
}

安装成功是上面这样子

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值