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