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 在這方面它只適合做客戶端,不適合做服務端。