PHP socket 的簡單理解

Luson發表於2019-09-13

php中的socket

socket建立套接的過程圖

PHP socket 的簡單理解

那麼socket是怎麼建立連線的呢?上面已經提到過了,它建立連線的過程是與mysql的客戶端和服務端的連線本質是一樣的。而它與mysql不同的是,mysql的服務端和客戶端都已經為我們編輯好了,我們只要應用就行了。但是,提供給我們的就是:幾十個socket函式。
下面介紹 socket 的函式:

  • socket的關鍵函式1:
          socket_create($net引數1,$stream引數2,$protocol引數3)
        作用:建立一個socket套接字,說白了,就是一個網路資料流。
        返回值:一個套接字,或者是false,引數錯誤發出E_WARNING警告
        php的線上手冊那裡說得更清楚:
        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-c...

    引數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)\

陣列裡面都是設定超時的最大時間,不過,一個是秒為單位,一個是微秒單位,作用都一樣

相關文章