php的socket編程算是比較難以理解的東西吧,不過,我們只要理解socket幾個函數之間的關係,以及它們所扮演的角色,那麼理解起來應該不是很難了,在筆者看來,socket編程,其實就是建立一個網路服務的用戶端和服務端,這和mysql的用戶端和服務端是一樣的,你只要理解mysql的用戶端和服務端是怎麼一回事,你就應該能夠理解下面我要講的東西吧。
關於socket編程所涉及到的網路通訊協定,什麼TCP啊,UDP啊,什麼socket三向交握等等,這些網路通訊協定網上有很詳細的解釋,這裡不講,只截個socket建立套接的過程圖讓你瞧瞧:
這個圖是我辛辛苦苦從別人那裡盜截過來的,你一定要好好看啊,同時,在這裡我也向那個被我盜截圖的筆者表示感謝,本人對自己盜取你的圖案表示歉意,還望你大人大量不要計較啊。本人實在太懶了,懶得畫圖,(其實是對自己的畫圖技術表示不自信,呵呵)。
socket是怎麼建立串連的呢?上面已經提到過了,它建立串連的過程是與mysql的用戶端和服務端的串連本質是一樣的。而它與mysql不同的是,mysql的服務端和用戶端都已經為我們編輯好了,我們只要應用就行了。但是,關鍵時刻來啦,socket它什麼東西都沒有提供給我們,唯一提供給我們的就是:幾十個socket函數。
這言外之意就是說,socket編程就是要我們自己建立服務端和用戶端,也就是說,``socket編程``——就是要我們自己建立一個類似於mysql的服務端和用戶端的應用。
說到這裡,我想問一句,你說這socket讓人頭疼不?它既不建立個服務端,也不建立個用戶端給我們應用,非要讓我們自己去應用socket的函數,建立一個屬於我們自己的網路通訊協定套接應用,這是不是很讓你頭疼呢?頭疼也沒辦法,要是你需要自己的應用,你還是不得不跟socket打交道。呵呵,這隻是題外話,不多說,下面進入正題。
在你沒有被socket編程搞蒙之前,我還是讓你看看socket的幾個關鍵函數,先給你解釋一下它們各自的作用。不然,要是對socket編程一點基礎都沒有的人看到了,我怕你看了之後,就果斷跳過這篇文章,從此對socket產生恐懼症了。呵呵,又多說了。
socket的關鍵函數1:
socket_create($net參數1,$stream參數2,$protocol參數3)
作用:建立一個socket通訊端,說白了,就是一個網路資料流。
傳回值:一個通訊端,或者是false,參數錯誤發出E_WARNING警告
php的線上手冊那裡說得更清楚:
socket_create建立並返回一個通訊端,也稱作一個通訊節點。一個典型的網路連接由 2 個通訊端構成,一個運行在用戶端,另一個運行在伺服器端。
上面一句話是從php線上手冊那裡複製過來的。看到沒有,這裡說得意思是不是和我上面反反覆複提到的用戶端與服務端一模一樣?呵呵。
參數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編程的時候,我覺得你還是得利用起來,特別是新手,可以當做調試用
下面就是代碼了,注意注意,請認真看我的注釋,注釋很重要,注釋很重要,注釋很重要,重要的事情要大喊三遍,呵呵。
服務端指令碼,D:\vhost\test\socket\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
用戶端指令碼,D:\vhost\test\socket\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 = 'l love you 我愛你 socket'; //轉為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);//工作完畢,關閉套接流
怎麼測試這兩個指令碼呢?
首先開啟windows的dos視窗,就是cmd黑視窗,然後,運行php D:\vhost\test\socket\server_socket.php,
讓服務端的的黑視窗持續啟動並執行,
其次,php的用戶端指令碼可以通過瀏覽器運行,也可以再開一個cmd黑視窗運行
php D:\vhost\test\socket\client_socket.php
在這裡請注意:php這個運行命名必須加入windows的環境變數中,假如不知道怎麼加,
請進入php運行命令目錄用絕對命令運行,也可以百度把php命令加入環境變數中
這裡是我的情況,你的檔案地址可能和我不一樣,請按照你的地址情況來操作,否則,後果自負,呵呵
上面已經說過了,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)
數組裡面都是設定逾時的最大時間,不過,一個是秒為單位,一個是微秒單位,作用都一樣