You've already forked qlg.tsgz.moe
							
							
		
			
				
	
	
		
			127 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			127 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
<?php
 | 
						|
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,
 | 
						|
            'max_request' => 0,
 | 
						|
        ));
 | 
						|
 | 
						|
        $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', 'www4swoole');
 | 
						|
            $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)
 | 
						|
    {
 | 
						|
	echo "Received: $data\n";
 | 
						|
        //没有空闲的数据库连接
 | 
						|
        
 | 
						|
	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();
 |