摘要:Socket是應用層與TCP/IP協議族通信的中間軟件抽象層,本文主要講PHP怎么連接socket,其中有幾個關鍵函數:socket_create() ,socket_connect(),socket_send...
什么是TCP/IP、UDP、Socket?
TCP/IP(Transmission Control Protocol/Internet Protocol)即傳輸控制協議/網間協議,是一個工業標準的協議集,它是為廣域網(WANs)設計的。
UDP(User Data Protocol,用戶數據報協議)是與TCP相對應的協議。它是屬于TCP/IP協議族中的一種。
TCP/IP協議族包括運輸層、網絡層、鏈路層。
Socket是應用層與TCP/IP協議族通信的中間軟件抽象層,它是一組接口。在設計模式中,Socket其實就是一個門面模式,它把復雜的TCP /IP協議族隱藏在Socket接口后面,對用戶來說,一組簡單的接口就是全部,讓Socket去組織數據,以符合指定的協議。
socket建立套接的過程圖:
首先了解socket 幾個關鍵函數:
關鍵函數1:
socket_create($net參數1,$stream參數2,$protocol參數3)
作用:創建一個socket套接字,說白了,就是一個網絡數據流。
返回值:一個套接字,或者是false,參數錯誤發出E_WARNING警告
socket_create創建并返回一個套接字,也稱作一個通訊節點。一個典型的網絡連接由 2 個套接字構成,一個運行在客戶端,另一個運行在服務器端。
參數1是:網絡協議,選項有:
AF_INET: IPv4 網絡協議。TCP 和 UDP 都可使用此協議。一般都用這個,你懂的。
AF_INET6: IPv6 網絡協議。TCP 和 UDP 都可使用此協議。
AF_UNIX: 本地通訊協議。具有高性能和低成本的 IPC(進程間通訊)。
參數2:套接字流,選項有:
SOCK_STREAM | SOCK_DGRAM | SOCK_SEQPACKET | SOCK_RAW | SOCK_RDM。
這里只對前兩個進行解釋:
SOCK_STREAM TCP 協議套接字。
SOCK_DGRAM UDP協議套接字。
欲了解更多請鏈接這里:http://php.net/manual/zh/function.socket-create.php
參數3:protocol協議,選項有:
SOL_TCP: TCP 協議。
SOL_UDP: UDP協議。
從這里可以看出,其實socket_create函數的第二個參數和第三個參數是相關聯的。
比如,假如你第一個參數應用IPv4協議:AF_INET,然后,第二個參數應用的是TCP套接字:SOCK_STREAM,
那么第三個參數必須要用 SOL_TCP ,這個應該不難理解。
TCP 協議套接字嘛,當然只能用TCP協議了,是不是?如果你應用UDP套接字,那么第三個參數該怎么選擇我就不說了,呵呵,你懂的。
關鍵函數2:
socket_connect($socket參數1,$ip參數2,$port參數3)
作用:連接一個套接字,返回值為true或者false
參數1:socket_create的函數返回值
參數2:ip地址
參數3:端口號
關鍵函數3:
socket_bind($socket參數1,$ip參數2,$port參數3)
作用:綁定一個套接字,返回值為true或者false
參數1:socket_create的函數返回值
參數2:ip地址
參數3:端口號
關鍵函數4:
socket_listen($socket參數1,$backlog 參數2)
作用:監聽一個套接字,返回值為true或者false
參數1:socket_create的函數返回值
參數2:最大監聽套接字個數
關鍵函數5:
socket_accept($socket)
作用:接收套接字的資源信息,成功返回套接字的信息資源,失敗為false
參數:socket_create的函數返回值
關鍵函數6:
socket_read($socket參數1,$length參數2)
作用:讀取套接字的資源信息,
返回值:成功把套接字的資源轉化為字符串信息,失敗為false
參數1:socket_create或者socket_accept的函數返回值
參數2:讀取的字符串的長度
關鍵函數7:
socket_write($socket參數1,$msg參數2,$strlen參數3)
作用:把數據寫入套接字中
返回值:成功返回字符串的字節長度,失敗為false
參數1:socket_create或者socket_accept的函數返回值
參數2:字符串
參數3:字符串的長度
關鍵函數8:
socket_close($socket)
作用:關閉套接字
返回值:成功返回true,失敗為false
參數:socket_create或者socket_accept的函數返回值
這八個函數是socket的核心函數,下面列舉幾個比較重要的函數:
socket_last_error($socket)
參數為socket_create的返回值,作用是獲取套接字的最后一條錯誤碼號,返回值套接字code
socket_strerror($code)
參數為socket_last_error函數的返回值,獲取code的字符串信息,返回值也就是套接字的錯誤信息
注:這兩個函數在socket編程中還是很重要的,在寫socket編程的時候,我覺得你還是得利用起來,特別是新手,可以當做調試用
socket_set_option($socket參數1 ,$level 參數2,$optname 參數3,$optval 參數4)
這個函數的作用是給套接字設置數據流選項,還是一個很重要的函數。
參數1:socket_create或者socket_accept的函數返回值
參數2:SOL_SOCKET,好像只有這個選項
參數3與參數4是相關聯的,
參數3可為:SO_REUSEADDR | SO_RCVTIMEO | S0_SNDTIMEO
解釋一下:
SO_REUSEADDR 是讓套接字端口釋放后立即就可以被再次使用
參數3假如是這個,則參數4可以為true或者false
SO_RCVTIMEO 是套接字的接收資源的最大超時時間
SO_SNDTIMEO 是套接字的發送資源的最大超時時間
參數3假如是后兩個,則參數4是一個這樣的數組array('sec'=>1,'usec'=>500000)
數組里面都是設置超時的最大時間,不過,一個是秒為單位,一個是微秒單位,作用都一樣
示例代碼:
服務端腳本 server_socket.php:
<?php //創建服務端的socket套接流,net協議為IPv4,protocol協議為TCP $socket = socket_create(AF_INET,SOCK_STREAM,SOL_TCP); /*綁定接收的套接流主機和端口,與客戶端相對應;端口可以自定義,前提是沒有被占用*/ if(socket_bind($socket,'127.0.0.1',8888) == false){ echo 'server bind fail:'.socket_strerror(socket_last_error()); /*這里的127.0.0.1是在本地主機測試,你如果有多臺電腦,可以寫IP地址*/ } //監聽套接流 if(socket_listen($socket,4)==false){ echo 'server listen fail:'.socket_strerror(socket_last_error()); } //讓服務器無限獲取客戶端傳過來的信息 do{ /*接收客戶端傳過來的信息*/ $accept_resource = socket_accept($socket); /*socket_accept的作用就是接受socket_bind()所綁定的主機發過來的套接流*/ if($accept_resource !== false){ /*讀取客戶端傳過來的資源,并轉化為字符串*/ $string = socket_read($accept_resource,1024); /*socket_read的作用就是讀出socket_accept()的資源并把它轉化為字符串*/ echo 'server receive is :'.$string.PHP_EOL;//PHP_EOL為php的換行預定義常量 if($string != false){ $return_client = 'server receive is : '.$string.PHP_EOL; /*向socket_accept的套接流寫入信息,也就是回饋信息給socket_bind()所綁定的主機客戶端*/ socket_write($accept_resource,$return_client,strlen($return_client)); /*socket_write的作用是向socket_create的套接流寫入信息,或者向socket_accept的套接流寫入信息*/ }else{ echo 'socket_read is fail'; } /*socket_close的作用是關閉socket_create()或者socket_accept()所建立的套接流*/ socket_close($accept_resource); } }while(true); socket_close($socket);
小提示:請注意上面的socket_bind,socket_listen,socket_accept三個函數的執行順序不可更改,也就是說必須先執行socket_bind,再執行socket_listen,最后才執行socket_accept.
客戶端腳本 client_socket.php:
<?php //創建一個socket套接流 $socket = socket_create(AF_INET,SOCK_STREAM,SOL_TCP); /****************設置socket連接選項,這兩個步驟你可以省略*************/ //接收套接流的最大超時時間1秒,后面是微秒單位超時時間,設置為零,表示不管它 socket_set_option($socket, SOL_SOCKET, SO_RCVTIMEO, array("sec" => 1, "usec" => 0)); //發送套接流的最大超時時間為6秒 socket_set_option($socket, SOL_SOCKET, SO_SNDTIMEO, array("sec" => 6, "usec" => 0)); /****************設置socket連接選項,這兩個步驟你可以省略*************/ //連接服務端的套接流,這一步就是使客戶端與服務器端的套接流建立聯系 if(socket_connect($socket,'127.0.0.1',8888) == false){ echo 'connect fail massege:'.socket_strerror(socket_last_error()); }else{ $message = 'YzmCMS輕量級開源CMS'; //轉為GBK編碼,處理亂碼問題,這要看你的編碼情況而定,每個人的編碼都不同 $message = mb_convert_encoding($message,'GBK','UTF-8'); //向服務端寫入字符串信息 if(socket_write($socket,$message,strlen($message)) == false){ echo 'fail to write'.socket_strerror(socket_last_error()); }else{ echo 'client write success'.PHP_EOL; //讀取服務端返回來的套接流信息 while($callback = socket_read($socket,1024)){ echo 'server return message is:'.PHP_EOL.$callback; } } } socket_close($socket);//工作完畢,關閉套接流