加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 站长学院 > PHP教程 > 正文

PHP中的socket_read和socket_recv区别详解

发布时间:2020-12-13 02:24:11 所属栏目:PHP教程 来源:网络整理
导读:《PHP实战:PHP中的socket_read和socket_recv区别详解》要点: 本文介绍了PHP实战:PHP中的socket_read和socket_recv区别详解,希望对您有用。如果有疑问,可以联系我们。 前几天用PHP写一个socket网络服务,在文档里看到socket_read和socket_recv这两个办法

《PHP实战:PHP中的socket_read和socket_recv区别详解》要点:
本文介绍了PHP实战: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的源码吧!
代码如下:

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 *,-1,le_socket_name,le_socket);

??? /* overflow check */
??? if ((len + 1) < 2) {
??????? RETURN_FALSE;
??? }

??? recv_buf = emalloc(len + 1);
??? memset(recv_buf,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] = '';

??????? /* 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,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,我们来看下源码:
代码如下:

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

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

*t = '';
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了关键部分,可以看到,这里的实现是一直循环调用recv,直到遇到r或者n或者读的数据长度到了指定的maxlen.

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

编程之家培训学院每天发布《PHP实战:PHP中的socket_read和socket_recv区别详解》等实战技能,PHP、MYSQL、LINUX、APP、JS,CSS全面培养人才。

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读