• 欢迎访问搞代码网站,推荐使用最新版火狐浏览器和Chrome浏览器访问本网站!
  • 如果您觉得本站非常有看点,那么赶紧使用Ctrl+D 收藏搞代码吧

PHP中的socket_read和socket_recv区别详解_php实例

php 搞代码 3年前 (2022-01-25) 13次浏览 已收录 0个评论

前几天用PHP写一个socket网络服务,在文档里看到socket_read和socket_recv这两个方法时有点晕,乍一看这不是一样的嘛,干吗还要给两个不同的用法呢。看文档没看太明白,看了下源码才搞清楚,在这里记录一下。

先看一下这两个函数的声明:

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

可以看到,从声明可以看到,一个是把收到的数据通过执行结果返回,另一个是把收到的数据通过引用的形式返回。另一个区别就是,socket_read多了一个type,socket_recv多了一个flags(够混乱的)。我们先来看看socket_recv的源码吧!

<br />PHP_FUNCTION(socket_recv)<br />{<br />    zval        *php_sock_res, *buf;<br />    char        *recv_buf;<br />    php_socket  *php_sock;<br />    int         retval;<br />    long        len, flags;</p><p>    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rzll", &php_sock_res, &buf, &len, &flags) == FAILURE) {<br />        return;<br />    }</p><p>    ZEND_FETCH_RESOURCE(php_sock, php_socket *, &php_sock_res, -1, le_socket_name, le_socket);</p><p>    /* overflow check */<br />    if ((len + 1) < 2) {<br />        RETURN_FALSE;<br />    }</p><p>    recv_buf = emalloc(len + 1);<br />    memset(recv_buf, 0, len + 1);</p><p>    if ((retval = recv(php_sock->bsd_socket, recv_buf, len, flags)) < 1) {<br />        efree(recv_buf);</p><p>        zval_dtor(buf);<br />        Z_TYPE_P(buf) = IS_NULL;<br />    } else {<br />        recv_buf[retval] = '\0';</p><p>        /* Rebuild buffer zval */<br />        zval_dtor(buf);</p><p>        Z_STRVAL_P(buf) = recv_buf;<br />        Z_STRLEN_P(buf) = retval;<br />        Z_TYPE_P(buf) = IS_STRING;<br />    }</p><p>    if (retval == -1) {<br />        PHP_SOCKET_ERROR(php_sock, "unable to read from socket", errno);<br />        RETURN_FALSE;<br />    }</p><p>    RETURN_LONG(retval);<br />}<br />

啰里啰嗦一大堆,其实有一行最关键:

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

可以看到,实际上这个函数就是调用了系统的recv而已,只是把输入参数和得到的结果都处理了一下,比较好理解。那我们再来看下socket_read,socket_read比系统的recv函数多了一个$type参数,这也是我认为这个函数存在的意义,从文档里可以看到,type有两个值,分别是PHP_BINARY_READ和PHP_NORMAL_READ,文档里有写,PHP_BINARY_READ表示直接用系统的recv方法,PHP_NORMAL_READ表示会一读,直到遇到\n 或者 \r,我们来看下源码:

<br />//省略一大堆<br />if (type == PHP_NORMAL_READ) {<br />    retval = php_read(php_sock, tmpbuf, length, 0);<br />} else {<br />    retval = recv(php_sock->bsd_socket, tmpbuf, length, 0);<br />}<br />

可以看到,如果是PHP_NORMAL_READ模式,其实行为和socket_recv是一样的,都是用的系统的recv函数,但是如果是PHP_NORMAL_READ,则有很大区别,用了自己实现的php_read函数,那这个php_read是干啥的呢?我们继续看源码:

<br />*t = '\0';<br />while (*t != '\n' && *t != '\r' && n < maxlen) {<br />    if (m > 0) {<br />        t++;<br />        n++;<br />    } else if (m == 0) {<br />        no_read++;<br />        if (nonblock && no_read >= 2) {<br />            return n;<br />            /* The first pass, m always is 0, so no_read becomes 1<br />             * in the first pass. no_read becomes 2 in the second pass,<br />             * and if this is nonblocking, we should return.. */<br />        }</p><p>        if (no_read > 200) {<br />            set_errno(ECONNRESET);<br />            return -1;<br />        }<br />    }</p><p>    if (n < maxlen) {<br />        m = recv(sock->bsd_socket, (void *) t, 1, flags);<br />    }</p><p>    if (errno != 0 && errno != ESPIPE && errno != E<a>@本文9来源gao($daima.com搞@代@#码8网^</a><strong>搞代gaodaima码</strong>AGAIN) {<br />        return -1;<br />    }</p><p>    set_errno(0);<br />}<br />

还是指copy了关键部分,可以看到,这里的实现是一直循环调用recv,直到遇到\r或者\n或者读的数据长度到了指定的maxlen。

虽然这两个函数比较混乱,但是看到这里应该明白了吧!好了睡觉去啦!


搞代码网(gaodaima.com)提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发送到邮箱[email protected],我们会在看到邮件的第一时间内为您处理,或直接联系QQ:872152909。本网站采用BY-NC-SA协议进行授权
转载请注明原文链接:PHP中的socket_read和socket_recv区别详解_php实例

喜欢 (0)
[搞代码]
分享 (0)
发表我的评论
取消评论

表情 贴图 加粗 删除线 居中 斜体 签到

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址