国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 語言 > PHP > 正文

PHP中的socket_read和socket_recv區別詳解

2024-09-04 11:42:52
字體:
來源:轉載
供稿:網友

這篇文章主要介紹了PHP中的socket_read和socket_recv區別詳解,本文從源碼上分析了這兩個函數的不同之處,需要的朋友可以參考下

前幾天用PHP寫一個socket網絡服務,在文檔里看到socket_read和socket_recv這兩個方法時有點暈,乍一看這不是一樣的嘛,干嗎還要給兩個不同的用法呢。看文檔沒看太明白,看了下源碼才搞清楚,在這里記錄一下。

先看一下這兩個函數的聲明:

string socket_read ( resource $socket , int $length [, int $type = PHP_BINARY_READ ] )

int socket_recv ( resource $socket , string &$buf , int $len , int $flags )

可以看到,從聲明可以看到,一個是把收到的數據通過執行結果返回,另一個是把收到的數據通過引用的形式返回。另一個區別就是,socket_read多了一個type,socket_recv多了一個flags(夠混亂的)。我們先來看看socket_recv的源碼吧!代碼如下:

  1. PHP_FUNCTION(socket_recv) 
  2.     zval        *php_sock_res, *buf; 
  3.     char        *recv_buf; 
  4.     php_socket  *php_sock; 
  5.     int         retval; 
  6.     long        len, flags; 
  7.     if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rzll", &php_sock_res, &buf, &len, &flags) == FAILURE) { 
  8.         return
  9.     } 
  10.  
  11.     ZEND_FETCH_RESOURCE(php_sock, php_socket *, &php_sock_res, -1, le_socket_name, le_socket); 
  12.  
  13.     /* overflow check */ 
  14.     if ((len + 1) < 2) { 
  15.         RETURN_FALSE; 
  16.     } 
  17.  
  18.     recv_buf = emalloc(len + 1); 
  19.     memset(recv_buf, 0, len + 1); 
  20.  
  21.     if ((retval = recv(php_sock->bsd_socket, recv_buf, len, flags)) < 1) { 
  22.         efree(recv_buf); 
  23.  
  24.         zval_dtor(buf); 
  25.         Z_TYPE_P(buf) = IS_NULL
  26.     } else { 
  27.         recv_buf[retval] = '/0'
  28.  
  29.         /* Rebuild buffer zval */ 
  30.         zval_dtor(buf); 
  31.  
  32.         Z_STRVAL_P(buf) = recv_buf; 
  33.         Z_STRLEN_P(buf) = retval; 
  34.         Z_TYPE_P(buf) = IS_STRING
  35.     } 
  36.  
  37.     if (retval == -1) { 
  38.         PHP_SOCKET_ERROR(php_sock, "unable to read from socket", errno); 
  39.         RETURN_FALSE; 
  40.     } 
  41.  
  42.     RETURN_LONG(retval); 

啰里啰嗦一大堆,其實有一行最關鍵:

if ((retval = recv(php_sock->bsd_socket, recv_buf, len, flags)) < 1) {

可以看到,實際上這個函數就是調用了系統的recv而已,只是把輸入參數和得到的結果都處理了一下,比較好理解。那我們再來看下socket_read,socket_read比系統的recv函數多了一個$type參數,這也是我認為這個函數存在的意義,從文檔里可以看到,type有兩個值,分別是PHP_BINARY_READ和PHP_NORMAL_READ,文檔里有寫,PHP_BINARY_READ表示直接用系統的recv方法,PHP_NORMAL_READ表示會一讀,直到遇到/n 或者 /r,我們來看下源碼:

  1. //省略一大堆 
  2. if (type == PHP_NORMAL_READ) { 
  3.     retval = php_read(php_sock, tmpbuf, length, 0); 
  4. else { 
  5.     retval = recv(php_sock->bsd_socket, tmpbuf, length, 0); 

可以看到,如果是PHP_NORMAL_READ模式,其實行為和socket_recv是一樣的,都是用的系統的recv函數,但是如果是PHP_NORMAL_READ,則有很大區別,用了自己實現的php_read函數,那這個php_read是干啥的呢?我們繼續看源碼:

  1. *t = '/0'
  2. while (*t != '/n' && *t != '/r' && n < maxlen) { 
  3.     if (m > 0) { 
  4.         t++; 
  5.         n++; 
  6.     } else if (m == 0) { 
  7.         no_read++; 
  8.         if (nonblock && no_read >= 2) { 
  9.             return n; 
  10.             /* The first pass, m always is 0, so no_read becomes 1 
  11.              * in the first pass. no_read becomes 2 in the second pass, 
  12.              * and if this is nonblocking, we should return.. */ 
  13.         } 
  14.         if (no_read > 200) { 
  15.             set_errno(ECONNRESET); 
  16.             return -1; 
  17.         } 
  18.     } 
  19.  
  20.     if (n < maxlen) { 
  21.         m = recv(sock->bsd_socket, (void *) t, 1, flags); 
  22.     } 
  23.  
  24.     if (errno != 0 && errno != ESPIPE && errno != EAGAIN) { 
  25.         return -1; 
  26.     } 
  27.  
  28.     set_errno(0); 

還是指copy了關鍵部分,可以看到,這里的實現是一直循環調用recv,直到遇到/r或者/n或者讀的數據長度到了指定的maxlen。

雖然這兩個函數比較混亂,但是看到這里應該明白了吧!好了睡覺去啦!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 平遥县| 五华县| 黑河市| 儋州市| 长宁区| 登封市| 保定市| 娱乐| 行唐县| 邹城市| 阳朔县| 青川县| 马关县| 衡南县| 南召县| 霍邱县| 信阳市| 新河县| 平顺县| 淮北市| 综艺| 太保市| 新巴尔虎右旗| 荥阳市| 赤壁市| 黄骅市| 忻州市| 那曲县| 阳新县| 桃江县| 关岭| 全椒县| 新巴尔虎左旗| 张家口市| 新化县| 兴隆县| 肇州县| 保亭| 安化县| 鲁甸县| 铜陵市|