php的socket通訊

奮程式序猿發表於2016-11-30

 socket通常叫做‘套接字’,用於描述IP地址和埠,是一個通訊鏈的控制程式碼。應用程式通過套接字向網路發出請求或者應答忘了請求。socket既不是程式,也不是協議,其只是作業系統提供的通訊層的一組抽象API。

通訊需要服務端和客戶端組成:

服務端:

使用php初始化socket然後繫結一個埠,對埠進行監聽。呼叫accept阻塞,等待客戶端連線。

客戶端:

客戶端初始化一個socket,然後連線伺服器,如果連線成功,這時客戶端與伺服器端的連線就建立了。客戶端傳送資料請求,伺服器端接收請求並處理請求,然後把迴應資料傳送給客戶端,客戶端讀取資料,最後關閉連線,一次互動結束。

socket相關函式:

socket_accept() 接受一個Socket連線
socket_bind() 把socket繫結在一個IP地址和埠上
socket_clear_error() 清除socket的錯誤或者最後的錯誤程式碼
socket_close() 關閉一個socket資源
socket_connect() 開始一個socket連線
socket_create_listen() 在指定埠開啟一個socket監聽
socket_create_pair() 產生一對沒有區別的socket到一個陣列裡
socket_create() 產生一個socket,相當於產生一個socket的資料結構
socket_get_option() 獲取socket選項
socket_getpeername() 獲取遠端類似主機的ip地址
socket_getsockname() 獲取本地socket的ip地址
socket_iovec_add() 新增一個新的向量到一個分散/聚合的陣列
socket_iovec_alloc() 這個函式建立一個能夠傳送接收讀寫的iovec資料結構
socket_iovec_delete() 刪除一個已經分配的iovec
socket_iovec_fetch() 返回指定的iovec資源的資料
socket_iovec_free() 釋放一個iovec資源
socket_iovec_set() 設定iovec的資料新值
socket_last_error() 獲取當前socket的最後錯誤程式碼
socket_listen() 監聽由指定socket的所有連線
socket_read() 讀取指定長度的資料
socket_readv() 讀取從分散/聚合陣列過來的資料
socket_recv() 從socket裡結束資料到快取
socket_recvfrom() 接受資料從指定的socket,如果沒有指定則預設當前socket
socket_recvmsg() 從iovec裡接受訊息
socket_select() 多路選擇
socket_send() 這個函式傳送資料到已連線的socket
socket_sendmsg() 傳送訊息到socket
socket_sendto() 傳送訊息到指定地址的socket
socket_set_block() 在socket裡設定為塊模式
socket_set_nonblock() socket裡設定為非塊模式
socket_set_option() 設定socket選項
socket_shutdown() 這個函式允許你關閉讀、寫、或者指定的socket
socket_strerror() 返回指定錯誤號的詳細錯誤
socket_write() 寫資料到socket快取
socket_writev() 寫資料到分散/聚合陣列

 

案例:

服務端

 

<?php
set_time_limit(0); //限制執行時間  0為不限制
$ip = '127.0.0.1';
$port = 8001;//

/**
socket通訊整個過程 
socket_create  //建立一個套接字
socket_bind  //給套接字繫結 ip 和埠
socket_listen //監聽套接字上的連線
socket_accept //接受一個socket連線
socket_read //接收客戶端 傳送的資料
socket_write //將資料寫到 socket 快取 向客戶端傳送
socket_close   //關閉套接字資源
*/ 

if(($sock = socket_create(AF_INET,SOCK_STREAM,SOL_TCP)) < 0) {
    echo "socket_create() 失敗的原因是:".socket_strerror($sock)."\n";
}

if(($ret = socket_bind($sock,$ip,$port)) < 0) {
    echo "socket_bind() 失敗的原因是:".socket_strerror($ret)."\n";
}

if(($ret = socket_listen($sock,4)) < 0) {
    echo "socket_listen() 失敗的原因是:".socket_strerror($ret)."\n";
}

$count = 0;

do {
    if (($msgsock = socket_accept($sock)) < 0) {
        echo "socket_accept() failed: reason: " . socket_strerror($msgsock) . "\n";
        break;
    } else {
        
        //發到客戶端
        $msg ="測試成功!\n";
        socket_write($msgsock, $msg, strlen($msg));
        
        echo "測試成功了啊\n";
        $buf = socket_read($msgsock,8192);
        
        
        $talkback = "收到的資訊:$buf\n";
        echo $talkback;
        
        if(++$count >= 5){
            break;
        };
        
    
    }
    //echo $buf;
    socket_close($msgsock);

} while (true);

socket_close($sock);
?>

執行php 檔案

執行後 應該看不見結果  可以使用 

netstat -ntlp 檢視 8001 埠是否被佔用

 

 客戶端:

<?php
error_reporting(E_ALL);
set_time_limit(0);
echo "socket通訊客戶端\n";
$port = 8001;//
$ip = "127.0.0.1";//ip

/**
socket連線整個過程
 socket_create //建立一個socket 連線
 socket_connect // 開始一個socket連線  連線服務端
 socket_write //將資料寫入快取   向服務端傳送
 socket_read// 讀取服務端的結果
 socket_close // 關閉套接字資源
  */

$socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
if ($socket < 0) {
    echo "socket_create() failed: reason: " . socket_strerror($socket) . "\n";
}else {
    echo "OK.\n";
}

echo "試圖連線 '$ip' 埠 '$port'...\n";
$result = socket_connect($socket, $ip, $port);
if ($result < 0) {
    echo "socket_connect() failed.\nReason: ($result) " . socket_strerror($result) . "\n";
}else {
    echo "連線OK\n";
}

$in = "Ho\r\nfirst blood\r\n";
$out = '';

if(!socket_write($socket, $in, strlen($in))) {
    echo "socket_write() failed: reason: " . socket_strerror($socket) . "\n";
}else {
    echo "傳送到伺服器資訊成功!\n";
    echo "傳送的內容為:<font color='red'>$in</font> <br>";
}

while($out = socket_read($socket, 8192)) {
    echo "接收伺服器回傳資訊成功!\n";
    echo "接受的內容為:",$out;
}


echo "關閉SOCKET...\n";
socket_close($socket);
echo "關閉OK\n";
?>

執行客戶端得到結果:

 

在看看服務端的視窗結果:

 PHP 語言的特性決定了php 在這方面它只適合做客戶端,不適合做服務端。

相關文章