欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

PHP中的socket_read和socket_recv區(qū)別詳解

 更新時(shí)間:2015年02月09日 10:12:20   投稿:junjie  
這篇文章主要介紹了PHP中的socket_read和socket_recv區(qū)別詳解,本文從源碼上分析了這兩個(gè)函數(shù)的不同之處,需要的朋友可以參考下

前幾天用PHP寫一個(gè)socket網(wǎng)絡(luò)服務(wù),在文檔里看到socket_read和socket_recv這兩個(gè)方法時(shí)有點(diǎn)暈,乍一看這不是一樣的嘛,干嗎還要給兩個(gè)不同的用法呢??次臋n沒看太明白,看了下源碼才搞清楚,在這里記錄一下。

先看一下這兩個(gè)函數(shù)的聲明:

復(fù)制代碼 代碼如下:

string socket_read ( resource $socket , int $length [, int $type = PHP_BINARY_READ ] )
int socket_recv ( resource $socket , string &$buf , int $len , int $flags )

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

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);
}

啰里啰嗦一大堆,其實(shí)有一行最關(guān)鍵:

復(fù)制代碼 代碼如下:

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

可以看到,實(shí)際上這個(gè)函數(shù)就是調(diào)用了系統(tǒng)的recv而已,只是把輸入?yún)?shù)和得到的結(jié)果都處理了一下,比較好理解。那我們?cè)賮砜聪聅ocket_read,socket_read比系統(tǒng)的recv函數(shù)多了一個(gè)$type參數(shù),這也是我認(rèn)為這個(gè)函數(shù)存在的意義,從文檔里可以看到,type有兩個(gè)值,分別是PHP_BINARY_READ和PHP_NORMAL_READ,文檔里有寫,PHP_BINARY_READ表示直接用系統(tǒng)的recv方法,PHP_NORMAL_READ表示會(huì)一讀,直到遇到\n 或者 \r,我們來看下源碼:
復(fù)制代碼 代碼如下:

//省略一大堆
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模式,其實(shí)行為和socket_recv是一樣的,都是用的系統(tǒng)的recv函數(shù),但是如果是PHP_NORMAL_READ,則有很大區(qū)別,用了自己實(shí)現(xiàn)的php_read函數(shù),那這個(gè)php_read是干啥的呢?我們繼續(xù)看源碼:
復(fù)制代碼 代碼如下:

*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了關(guān)鍵部分,可以看到,這里的實(shí)現(xiàn)是一直循環(huán)調(diào)用recv,直到遇到\r或者\(yùn)n或者讀的數(shù)據(jù)長度到了指定的maxlen。

雖然這兩個(gè)函數(shù)比較混亂,但是看到這里應(yīng)該明白了吧!好了睡覺去啦!

相關(guān)文章

  • PHP超全局?jǐn)?shù)組(Superglobals)介紹

    PHP超全局?jǐn)?shù)組(Superglobals)介紹

    這篇文章主要介紹了PHP超全局?jǐn)?shù)組(Superglobals)介紹,本文講解了概述、變量的作用域、超全局?jǐn)?shù)組及注意事項(xiàng)等內(nèi)容,需要的朋友可以參考下
    2015-07-07
  • Zend Framework框架db類的分頁示例分享

    Zend Framework框架db類的分頁示例分享

    這篇文章主要介紹了Zend Framework框架db類的分頁示例,代碼很簡(jiǎn)單,大家看一下注釋就可以使用了
    2014-03-03
  • 全新的PDO數(shù)據(jù)庫操作類php版(僅適用Mysql)

    全新的PDO數(shù)據(jù)庫操作類php版(僅適用Mysql)

    在公司里也用了1年之久。如今公司規(guī)模變大了,產(chǎn)品也日益完善,曾經(jīng)的那個(gè)數(shù)據(jù)庫操作函數(shù)雖說使用上沒出什么大問題,但為了更顯專業(yè),花了1天時(shí)間重寫了這個(gè),現(xiàn)在,它確實(shí)是個(gè)類了
    2012-07-07
  • Laravel中基于Artisan View擴(kuò)展包創(chuàng)建及刪除應(yīng)用視圖文件的方法

    Laravel中基于Artisan View擴(kuò)展包創(chuàng)建及刪除應(yīng)用視圖文件的方法

    這篇文章主要介紹了Laravel中基于Artisan View擴(kuò)展包創(chuàng)建及刪除應(yīng)用視圖文件的方法,簡(jiǎn)單分析了Laravel擴(kuò)展包的安裝及視圖的創(chuàng)建與刪除操作相關(guān)技巧,需要的朋友可以參考下
    2016-10-10
  • 微信跳一跳php代碼實(shí)現(xiàn)

    微信跳一跳php代碼實(shí)現(xiàn)

    這篇文章主要為大家詳細(xì)介紹了微信跳一跳輔助php代碼實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • 使用PHP?Smarty處理表單數(shù)據(jù)的方法

    使用PHP?Smarty處理表單數(shù)據(jù)的方法

    這篇文章主要介紹了如何使用PHP?Smarty處理表單數(shù)據(jù),首先需要下載Smarty庫并將其解壓到你的項(xiàng)目,下面通過本文結(jié)合實(shí)例代碼給大家講解的非常詳細(xì),需要的朋友可以參考下
    2023-08-08
  • Codeigniter里的無刷新上傳的實(shí)現(xiàn)代碼

    Codeigniter里的無刷新上傳的實(shí)現(xiàn)代碼

    這篇文章主要介紹了Codeigniter里的無刷新上傳的實(shí)現(xiàn)代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • PHP兩種快速排序算法實(shí)例

    PHP兩種快速排序算法實(shí)例

    這篇文章主要介紹了PHP兩種快速排序算法實(shí)例,本文直接給出實(shí)現(xiàn)代碼,分別使用遞歸法、迭代法實(shí)現(xiàn),需要的朋友可以參考下
    2015-02-02
  • PHP網(wǎng)頁游戲?qū)W習(xí)之Xnova(ogame)源碼解讀(十)

    PHP網(wǎng)頁游戲?qū)W習(xí)之Xnova(ogame)源碼解讀(十)

    這篇文章主要介紹了PHP網(wǎng)頁游戲Xnova(ogame)源碼解讀的建造總覽部分,需要的朋友可以參考下
    2014-06-06
  • 六種php加密解密方法實(shí)例講解

    六種php加密解密方法實(shí)例講解

    這篇文章主要介紹了六種php加密解密方法實(shí)例講解,代碼講解的很清楚,有對(duì)這方面想要深入研究的同學(xué)可以看下
    2021-01-01

最新評(píng)論