PHP中的socket_read和socket_recv區(qū)別詳解
前幾天用PHP寫一個socket網(wǎng)絡服務,在文檔里看到socket_read和socket_recv這兩個方法時有點暈,乍一看這不是一樣的嘛,干嗎還要給兩個不同的用法呢??次臋n沒看太明白,看了下源碼才搞清楚,在這里記錄一下。
先看一下這兩個函數(shù)的聲明:
string socket_read ( resource $socket , int $length [, int $type = PHP_BINARY_READ ] )
int socket_recv ( resource $socket , string &$buf , int $len , int $flags )
可以看到,從聲明可以看到,一個是把收到的數(shù)據(jù)通過執(zhí)行結果返回,另一個是把收到的數(shù)據(jù)通過引用的形式返回。另一個區(qū)別就是,socket_read多了一個type,socket_recv多了一個flags(夠混亂的)。我們先來看看socket_recv的源碼吧!
PHP_FUNCTION(socket_recv)
{
zval *php_sock_res, *buf;
char *recv_buf;
php_socket *php_sock;
int retval;
long len, flags;
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rzll", &php_sock_res, &buf, &len, &flags) == FAILURE) {
return;
}
ZEND_FETCH_RESOURCE(php_sock, php_socket *, &php_sock_res, -1, le_socket_name, le_socket);
/* overflow check */
if ((len + 1) < 2) {
RETURN_FALSE;
}
recv_buf = emalloc(len + 1);
memset(recv_buf, 0, len + 1);
if ((retval = recv(php_sock->bsd_socket, recv_buf, len, flags)) < 1) {
efree(recv_buf);
zval_dtor(buf);
Z_TYPE_P(buf) = IS_NULL;
} else {
recv_buf[retval] = '\0';
/* Rebuild buffer zval */
zval_dtor(buf);
Z_STRVAL_P(buf) = recv_buf;
Z_STRLEN_P(buf) = retval;
Z_TYPE_P(buf) = IS_STRING;
}
if (retval == -1) {
PHP_SOCKET_ERROR(php_sock, "unable to read from socket", errno);
RETURN_FALSE;
}
RETURN_LONG(retval);
}
啰里啰嗦一大堆,其實有一行最關鍵:
if ((retval = recv(php_sock->bsd_socket, recv_buf, len, flags)) < 1) {
可以看到,實際上這個函數(shù)就是調(diào)用了系統(tǒng)的recv而已,只是把輸入?yún)?shù)和得到的結果都處理了一下,比較好理解。那我們再來看下socket_read,socket_read比系統(tǒng)的recv函數(shù)多了一個$type參數(shù),這也是我認為這個函數(shù)存在的意義,從文檔里可以看到,type有兩個值,分別是PHP_BINARY_READ和PHP_NORMAL_READ,文檔里有寫,PHP_BINARY_READ表示直接用系統(tǒng)的recv方法,PHP_NORMAL_READ表示會一讀,直到遇到\n 或者 \r,我們來看下源碼:
//省略一大堆
if (type == PHP_NORMAL_READ) {
retval = php_read(php_sock, tmpbuf, length, 0);
} else {
retval = recv(php_sock->bsd_socket, tmpbuf, length, 0);
}
可以看到,如果是PHP_NORMAL_READ模式,其實行為和socket_recv是一樣的,都是用的系統(tǒng)的recv函數(shù),但是如果是PHP_NORMAL_READ,則有很大區(qū)別,用了自己實現(xiàn)的php_read函數(shù),那這個php_read是干啥的呢?我們繼續(xù)看源碼:
*t = '\0';
while (*t != '\n' && *t != '\r' && n < maxlen) {
if (m > 0) {
t++;
n++;
} else if (m == 0) {
no_read++;
if (nonblock && no_read >= 2) {
return n;
/* The first pass, m always is 0, so no_read becomes 1
* in the first pass. no_read becomes 2 in the second pass,
* and if this is nonblocking, we should return.. */
}
if (no_read > 200) {
set_errno(ECONNRESET);
return -1;
}
}
if (n < maxlen) {
m = recv(sock->bsd_socket, (void *) t, 1, flags);
}
if (errno != 0 && errno != ESPIPE && errno != EAGAIN) {
return -1;
}
set_errno(0);
}
還是指copy了關鍵部分,可以看到,這里的實現(xiàn)是一直循環(huán)調(diào)用recv,直到遇到\r或者\n或者讀的數(shù)據(jù)長度到了指定的maxlen。
雖然這兩個函數(shù)比較混亂,但是看到這里應該明白了吧!好了睡覺去啦!
- php使用websocket示例詳解
- php中使用Curl、socket、file_get_contents三種方法POST提交數(shù)據(jù)
- php獲取遠程圖片的兩種 CURL方式和sockets方式獲取遠程圖片
- php使用socket編程示例
- PHP實現(xiàn)Socket服務器的代碼
- php socket客戶端及服務器端應用實例
- php中curl、fsocket、file_get_content三個函數(shù)的使用比較
- 使用swoole擴展php websocket示例
- 深入淺出php socket編程
- 基于PHP Socket配置以及實例的詳細介紹
- PHP異步調(diào)用socket實現(xiàn)代碼
- php與java通過socket通信的實現(xiàn)代碼
- php socket方式提交的post詳解
- PHP中Socket連接及讀寫數(shù)據(jù)超時問題分析
相關文章
全新的PDO數(shù)據(jù)庫操作類php版(僅適用Mysql)
在公司里也用了1年之久。如今公司規(guī)模變大了,產(chǎn)品也日益完善,曾經(jīng)的那個數(shù)據(jù)庫操作函數(shù)雖說使用上沒出什么大問題,但為了更顯專業(yè),花了1天時間重寫了這個,現(xiàn)在,它確實是個類了2012-07-07Laravel中基于Artisan View擴展包創(chuàng)建及刪除應用視圖文件的方法
這篇文章主要介紹了Laravel中基于Artisan View擴展包創(chuàng)建及刪除應用視圖文件的方法,簡單分析了Laravel擴展包的安裝及視圖的創(chuàng)建與刪除操作相關技巧,需要的朋友可以參考下2016-10-10使用PHP?Smarty處理表單數(shù)據(jù)的方法
這篇文章主要介紹了如何使用PHP?Smarty處理表單數(shù)據(jù),首先需要下載Smarty庫并將其解壓到你的項目,下面通過本文結合實例代碼給大家講解的非常詳細,需要的朋友可以參考下2023-08-08PHP網(wǎng)頁游戲學習之Xnova(ogame)源碼解讀(十)
這篇文章主要介紹了PHP網(wǎng)頁游戲Xnova(ogame)源碼解讀的建造總覽部分,需要的朋友可以參考下2014-06-06