首頁  >  文章  >  後端開發  >  php的socket程式設計詳解

php的socket程式設計詳解

高洛峰
高洛峰原創
2016-12-12 09:41:191370瀏覽

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

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

php的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)

 〟〜 〟

    回傳值:一個套接字,或者是false,參數錯誤發出E_WARNING警告

    php的在線手冊那裡說得更清楚:

   php的在線手冊那裡說得更清楚:

   php的在線手冊那裡說得更清楚:

〜〔一個字 〜」也返回一個套接一個一個典型的網路連線由 2 個套接字構成,一個運行在客戶端,另一個運行在伺服器端。

上面一句話是從php線上手冊複製過來的。看到沒有,這裡說得意思是不是跟我上面反覆提到的客戶端跟服務端一模一樣?呵呵。

    參數1是:網路協議,

    網路協定有哪些?它的選擇項目就下面這三個:

    AF_INET:     IPv4 網路協定。 TCP 和 UDP 都可使用此協定。一般都用這個,你懂的。

    AF_INET6:   IPv6 網路協定。 TCP 和 UDP 都可使用此協定。

    AF_UNIX:      本地通訊協定。具有高效能和低成本的 IPC(進程間通訊)。

    參數2:套接字流,選項有:

    SOCK_STREAM  SOCK_DGRAM   SEQPACKET  SOCK_DGRAM   SEQPACKET  。

    這裡只對前兩者進行解釋:

    SOCK_STREAM  TCP 協定套接字。

    SOCK_DGRAM   UDP協定套接字。

    欲了解更多請連結這裡:http://php.net/manual/zh/function.socket-create.php

    參數3:protocol協定,選項有:    參數3:」 TC 選項有:有:H〜 〜 〜 >選項有:有:『

    SOL_UDP:  UDP協定。

    從這裡可以看出,其實socket_create函數的第二個參數和第三個參數是相關聯的。

    例如,假如你第一個參數應用IPv4協定:AF_INET,然後,第二個參數應用的是TCP套接字:SOCK_STREAM,

   接字那麼第三個參數必須理解這個難解。 🎜

    TCP 協定套接字嘛,當然只能用TCP協定了,是不是?如果你應用UDP套接字,那麼第三個參數該怎麼選擇我就不說了,呵呵,你懂的。

    關鍵函數2:

    socket_connect($socket參數1,$ip參數2,$port參數3)

🎀  〜〟〟〜為接一個接合為〜為〜,或值。  參數1:socket_create的函數傳回值

    參數2:ip位址

    參數3:連接埠號碼

       參數3:連接埠號碼

    、]$

    作用:綁定一個套接字,傳回值為true或false

       參數1:socket_create的函數回傳值

    參數2:ip :

    socket_listen($socket參數1,$backlog 參數2 )

    作用:監聽一個套接字,傳回值為true或false

    參數1:socket_create的函數回傳值 關鍵函數5:

    socket_accept($ socket)

    作用:接收套接字的資源訊息,成功傳回套接字的資訊資源,失敗為false

        參數:socket_create的函數回傳  socket_read($socket參數1, $length參數2)

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

    回傳值:成功把套接字的資源轉換為字串訊息,失敗為false 的值

    參數2:讀取的字串的長度

    關鍵函數7:

    socket 把樣資料寫入套接字中

    回傳值:成功傳回字串的位元組長度,失敗為false

      參數1:socket_create或socket_accept的函數傳回 參數3:字串的長度

    關鍵函數8 :

    socket_close($socket)

    作用:關閉套接字

  〠 或是socket_accept的函數回傳值

    這八個函數是socket的核心函數,以下列舉兩個比較重要的函數

    socket_last_error($socket),參數為socket_create的回傳值,作用是取得套接字的最後一個錯誤碼號,傳回值套接字『 ),參數為socket_last_error函數的回傳值,取得code的字串資訊,回傳值也就是套接字的錯誤訊息

    這兩個函數在socket程式設計還是很重要的,在寫socket程式設計的時候,我覺得你還是得利用起來,特別是新手,可以當做調試用

    

    以下就是代碼了,注意注意,請認真看我的註釋,註釋很重要,註釋很重要,註釋很重要,下面就是代碼了,注意注意,請認真看我的註釋,註釋很重要,註釋很重要,註釋很重要,重要的事情要注意大喊三遍,呵呵。

    服務端腳本,D:vhosttestsocketserver_socket.php 

<?php
//创建服务端的socket套接流,net协议为IPv4,protocol协议为TCP
$socket = socket_create(AF_INET,SOCK_STREAM,SOL_TCP);

  /*绑定接收的套接流主机和端口,与客户端相对应*/
  if(socket_bind($socket,&#39;127.0.0.1&#39;,8888) == false){
    echo &#39;server bind fail:&#39;.socket_strerror(socket_last_error());
    /*这里的127.0.0.1是在本地主机测试,你如果有多台电脑,可以写IP地址*/
  }
  //监听套接流
  if(socket_listen($socket,4)==false){
    echo &#39;server listen fail:&#39;.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 &#39;server receive is :&#39;.$string.PHP_EOL;//PHP_EOL为php的换行预定义常量
    if($string != false){
      $return_client = &#39;server receive is : &#39;.$string.PHP_EOL;
      /*向socket_accept的套接流写入信息,也就是回馈信息给socket_bind()所绑定的主机客户端*/
      socket_write($accept_resource,$return_client,strlen($return_client));
      /*socket_write的作用是向socket_create的套接流写入信息,或者向socket_accept的套接流写入信息*/
    }else{
      echo &#39;socket_read is fail&#39;;
    }
  /*socket_close的作用是关闭socket_create()或者socket_accept()所建立的套接流*/
    socket_close($accept_resource);
  }
}while(true);
socket_close($socket);

     小提示:請注意上面的socket_bind,sock〦  小〟 必須先執行socket_bind,再執行socket_listen,最後才執行socket_accept

    客戶端腳本,D:vhosttestsocketclient_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,&#39;127.0.0.1&#39;,8888) == false){
    echo &#39;connect fail massege:&#39;.socket_strerror(socket_last_error());
  }else{
    $message = &#39;l love you 我爱你 socket&#39;;
    //转为GBK编码,处理乱码问题,这要看你的编码情况而定,每个人的编码都不同
    $message = mb_convert_encoding($message,&#39;GBK&#39;,&#39;UTF-8&#39;);
    //向服务端写入字符串信息

    if(socket_write($socket,$message,strlen($message)) == false){
      echo &#39;fail to write&#39;.socket_strerror(socket_last_error());

    }else{
      echo &#39;client write success&#39;.PHP_EOL;
      //读取服务端返回来的套接流信息
      while($callback = socket_read($socket,1024)){
        echo &#39;server return message is:&#39;.PHP_EOL.$callback;
      }
    }
  }
  socket_close($socket);//工作完毕,关闭套接流

    怎麼測試這兩個腳本呢?

    首先打開windows的dos窗口,就是cmd黑窗口,然後,運行php D:vhosttestsocketserver_socket.php,

    讓服務端的的php,

    讓服務端的的php,

    讓服務端的的 我可以持續使用的, ,也可以再開一個cmd黑窗口運行

    php D:vhosttestsocketclient_socket.php

      在這裡請注意: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_REUSEADDRSm.一下:

SO_REUSEADDR  是讓套接字連接埠釋放後立即就可以被再次使用

        參數3假如是這個,則參數4可以為truefalVT🀎

SO_SNDTIMEO是套接字的發送資源的最大超時時間

  參數3假如是這兩個,則參數4是一個這樣的數組array('sec'=>1,'usec'=>500000)

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


陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn