簡單理解php的socket編程

https://www.cnblogs.com/loveyoume/p/6076101.html  原文php

php的socket編程算是比較難以理解的東西吧,不過,咱們只要理解socket幾個函數之間的關係,以及它們所扮演的角色,那麼理解起來應該不是很難了,在筆者看來,socket編程,其實就是創建一個網絡服務的客戶端和服務端,這和mysql的客戶端和服務端是同樣的,你只要理解mysql的客戶端和服務端是怎麼一回事,你就應該可以理解下面我要講的東西吧。html

    關於socket編程所涉及到的網絡協議,什麼TCP啊,UDP啊,什麼socket三次握手等等,這些網絡協議網上有很詳細的解釋,這裏不講,只截個socket創建套接的過程圖讓你瞧瞧:mysql

                  

 

    這個圖是我辛辛苦苦從別人那裏盜截過來的,你必定要好好看啊,同時,在這裏我也向那個被我盜截圖的筆者表示感謝,本人對本身盜取你的圖案表示歉意,還望你大人大量不要計較啊。本人實在太懶了,懶得畫圖,(實際上是對本身的畫圖技術表示不自信,呵呵)。sql

    socket是怎麼創建鏈接的呢?上面已經提到過了,它創建鏈接的過程是與mysql的客戶端和服務端的鏈接本質是同樣的。而它與mysql不一樣的是,mysql的服務端和客戶端都已經爲咱們編輯好了,咱們只要應用就好了。可是,關鍵時刻來啦,socket它什麼東西都沒有提供給咱們,惟一提供給咱們的就是:幾十個socket函數。編程

    這言外之意就是說,socket編程就是要咱們本身建立服務端和客戶端,也就是說,``socket編程``——就是要咱們本身創建一個相似於mysql的服務端和客戶端的應用。windows

    說到這裏,我想問一句,你說這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)

  數組裏面都是設置超時的最大時間,不過,一個是秒爲單位,一個是微秒單位,做用都同樣

相關文章
相關標籤/搜索