簡單理解php的socket編程

happyxyb 8年前發布 | 16K 次閱讀 Socket PHP PHP開發

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)

作用:讀取套接字的資源信息,

返回值:成功把套接字的資源轉化為字符串信息,失敗為 false

參數: socket_create 或者 socket_accept 的函數返回值

關鍵函數 7 :

socket_write($socket 參數 1 , $msg 參數 2 , $strlen 參數 3 )

作用:把數據寫入套接字中

返回值:成功返回字符串的字節長度,失敗為 false

參數: socket_create 或者 socket_accept 的函數返回值

關鍵函數 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);

客戶端腳本,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)

數組里面都是設置超時的最大時間,不過,一個是秒為單位,一個是微秒單位,作用都一樣

 

來自:http://www.cnblogs.com/loveyoume/p/6076101.html

 

 本文由用戶 happyxyb 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!