PHP開發非同步高效能的MySQLProxy 伺服器

來源:互聯網
上載者:User

標籤:

ySQL資料庫對每個用戶端串連都會分配一個線程,所以串連非常寶貴。開發一個非同步MySQLProxy 伺服器,PHP應用伺服器可以長串連到這台Server,既減輕MYSQL的串連壓力,又使PHP保持長串連減少connect/close的網路開銷。

此Server考慮到了設定了資料庫連接池尺寸,區分忙閑,mysqli斷線重連,並設定了負載保護。基於swoole擴充開發,io迴圈使用epoll,是全非同步非阻塞的,可以應對大量TCP串連。

程式的邏輯是:啟動時建立N個MySQL串連,收到用戶端發來的SQL後,分配1個MySQL串連,將SQL發往資料庫伺服器。然後等待資料庫返回查詢結果。當資料庫返回結果後,再發給對應的用戶端串連。

核心的資料結構是3個PHP數組。idle_pool是閒置資料庫連接,當有SQL請求時從idle_pool中移到busy_pool中。當數 據庫返回結果後從busy_pool中再移到idle_pool中,以供新的請求使用。當SQL請求到達時如果沒有閒置資料庫連接,那會自動加入到 wait_queue中。一旦有SQL完成操作,將自動從wait_queue中取出等待的請求進行處理。

如此迴圈使用。由於整個伺服器是非同步單進程單線程所以完全不需要鎖。而且是完全非同步,效率非常高。

當然本文的代碼,如果要用於生產環境,還需做更多的保護機制和壓力測試。在此僅拋磚引玉,提供一個解決問題的思路。

class DBServer{    protected $pool_size = 20;    protected $idle_pool = array(); //空閑串連    protected $busy_pool = array(); //工作串連    protected $wait_queue = array(); //等待的請求    protected $wait_queue_max = 100; //等待隊列的最大長度,超過後將拒絕新的請求    /**     * @var swoole_server     */    protected $serv;    function run()    {        $serv = new swoole_server("127.0.0.1", 9509);        $serv->set(array(            ‘worker_num‘ => 1,        ));        $serv->on(‘WorkerStart‘, array($this, ‘onStart‘));        //$serv->on(‘Connect‘, array($this, ‘onConnect‘));        $serv->on(‘Receive‘, array($this, ‘onReceive‘));        //$serv->on(‘Close‘, array($this, ‘onClose‘));        $serv->start();    }    function onStart($serv)    {        $this->serv = $serv;        for ($i = 0; $i < $this->pool_size; $i++) {            $db = new mysqli;            $db->connect(‘127.0.0.1‘, ‘root‘, ‘root‘, ‘test‘);            $db_sock = swoole_get_mysqli_sock($db);            swoole_event_add($db_sock, array($this, ‘onSQLReady‘));            $this->idle_pool[] = array(                ‘mysqli‘ => $db,                ‘db_sock‘ => $db_sock,                ‘fd‘ => 0,            );        }        echo "Server: start.Swoole version is [" . SWOOLE_VERSION . "]\n";    }    function onSQLReady($db_sock)    {        $db_res = $this->busy_pool[$db_sock];        $mysqli = $db_res[‘mysqli‘];        $fd = $db_res[‘fd‘];        echo __METHOD__ . ": client_sock=$fd|db_sock=$db_sock\n";        if ($result = $mysqli->reap_async_query()) {            $ret = var_export($result->fetch_all(MYSQLI_ASSOC), true) . "\n";            $this->serv->send($fd, $ret);            if (is_object($result)) {                mysqli_free_result($result);            }        } else {            $this->serv->send($fd, sprintf("MySQLi Error: %s\n", mysqli_error($mysqli)));        }        //release mysqli object        $this->idle_pool[] = $db_res;        unset($this->busy_pool[$db_sock]);        //這裡可以取出一個等待請求        if (count($this->wait_queue) > 0) {            $idle_n = count($this->idle_pool);            for ($i = 0; $i < $idle_n; $i++) {                $req = array_shift($this->wait_queue);                $this->doQuery($req[‘fd‘], $req[‘sql‘]);            }        }    }    function onReceive($serv, $fd, $from_id, $data)    {        //沒有閒置資料庫連接        if (count($this->idle_pool) == 0) {            //等待隊列未滿            if (count($this->wait_queue) < $this->wait_queue_max) {                $this->wait_queue[] = array(                    ‘fd‘ => $fd,                    ‘sql‘ => $data,                );            } else {                $this->serv->send($fd, "request too many, Please try again later.");            }        } else {            $this->doQuery($fd, $data);        }    }    function doQuery($fd, $sql)    {        //從空閑池中移除        $db = array_pop($this->idle_pool);        /**         * @var mysqli         */        $mysqli = $db[‘mysqli‘];        for ($i = 0; $i < 2; $i++) {            $result = $mysqli->query($sql, MYSQLI_ASYNC);            if ($result === false) {                if ($mysqli->errno == 2013 or $mysqli->errno == 2006) {                    $mysqli->close();                    $r = $mysqli->connect();                    if ($r === true) continue;                }            }            break;        }        $db[‘fd‘] = $fd;        //加入工作池中        $this->busy_pool[$db[‘db_sock‘]] = $db;    }}$server = new DBServer();$server->run();

來源:http://rango.swoole.com/archives/288

PHP開發非同步高效能的MySQLProxy 伺服器

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.