如何用PHP實現Socket伺服器

釋懷發表於2015-06-27
socket伺服器的工作方式是這樣的,不間斷地執行以等待客戶端的連線。一旦客戶端連線上了,伺服器就會將它新增到客戶名單中,然後開始等待來自客戶端的訊息。


不要走開,下面是完整的原始碼:
// Set time limit to indefinite execution
set_time_limit (0);

// Set the ip and port we will listen on
$address = 'localhost';
$port = 10000;
$max_clients = 10;

// Array that will hold client information
$client = Array();

// Create a TCP Stream socket
$sock = socket_create(AF_INET, SOCK_STREAM, 0);
// Bind the socket to an address/port
socket_bind($sock, $address, $port) or die('Could not bind to address');
// Start listening for connections
socket_listen($sock);

echo "Waiting for connections...\r\n";

// Loop continuously
while (true) {
        // Setup clients listen socket for reading
        $read[0] = $sock;
        for ($i = 0; $i < $max_clients; $i++) {
                if (isset($client[$i]['sock']))
                        $read[$i + 1] = $client[$i]['sock'];
        }
        // Set up a blocking call to socket_select()
        if (socket_select($read, $write = NULL, $except = NULL, $tv_sec = 5) < 1)
                continue;
        /* if a new connection is being made add it to the client array */
        if (in_array($sock, $read)) {
                for ($i = 0; $i < $max_clients; $i++) {
                        if (empty($client[$i]['sock'])) {
                                $client[$i]['sock'] = socket_accept($sock);
                                echo "New client connected $i\r\n";
                                break;
                        }
                        elseif ($i == $max_clients - 1)
                                echo "Too many clients...\r\n";
                }
        } // end if in_array

        // If a client is trying to write - handle it now
        for ($i = 0; $i < $max_clients; $i++) { // for each client
                if (isset($client[$i]['sock'])) {
                        if (in_array($client[$i]['sock'], $read)) {
                                $input = socket_read($client[$i]['sock'], 1024);
                                if ($input == null) {
                                        echo "Client disconnecting $i\r\n";
                                        // Zero length string meaning disconnected
                                        unset($client[$i]);
                                } else {
                                        echo "New input received $i\r\n";
                                        // send it to the other clients
                                        for ($j = 0; $j < $max_clients; $j++) {
                                                if (isset($client[$j]['sock']) && $j != $i) {
                                                        echo "Writing '$input' to client $j\r\n";
                                                        socket_write($client[$j]['sock'], $input, strlen($input));
                                                }
                                        }
                                        if ($input == 'exit') {
                                                // requested disconnect
                                                socket_close($client[$i]['sock']);
                                        }
                                }
                        } else {
                                echo "Client disconnected $i\r\n";
                                // Close the socket
                                socket_close($client[$i]['sock']);
                                unset($client[$i]);
                        }
                }
        }
} // end while
// Close the master sockets
socket_close($sock);
啊呀,乍一看這似乎是一個大工程,但是我們可以先將它分解為幾個較小的部分。第一部分是建立伺服器。Lines:2至20。
這部分程式碼設定了變數、地址、埠、最大客戶端和客戶端陣列。接下來建立socket並將其繫結到我們指定的地址和埠上。
下面我們要做的事情就是執行一個死迴圈(實際上我們是故意的!)。Lines:22至32。在這部分程式碼中我們做的第一步是設定 $read 陣列。此陣列包含所有客戶端的套接字和我們主伺服器的套接字。這個變數稍後會用於select語句:告訴PHP監聽來自這些客戶端的每一條訊息。

socket_select()的最後一個引數告訴我們的伺服器在返回值之前最多等待5秒鐘。如果它的返回值小於1,那麼就表示沒有收到任何資料,所以只需要返回迴圈頂部,繼續等待。

指令碼的下一個部分,是增加新的客戶端到陣列中。Lines:33至44。

將新的客戶端放置在列表的末尾。檢查以確保客戶端的數量沒有超過我們想要伺服器處理的數量。

下面要介紹的程式碼塊相當大,也是伺服器的主要部分。當客戶端將訊息傳送到伺服器時,就需要這塊程式碼挺身而出來處理。訊息可以是各種各樣的,斷開訊息、實際斷開——只要是伺服器需要處理的訊息。Lines:46至末尾。

程式碼迴圈通過每個客戶端並檢查是否收到來自於它們的訊息。如果是,獲取輸入的內容。根據輸入來檢查這是否是一個斷開訊息,如果是那就從陣列中刪除它們,反之,那它就是一個正常的訊息,那我們的伺服器再次通過所有客戶端,並一個一個寫資訊給他們,跳過傳送者。

好了,下面試試創造你自己的聊天伺服器吧!
相關閱讀
評論(1)

相關文章