首页 > 作文

PHP中的socket

更新时间:2023-04-06 09:18:33 阅读: 评论:0

前几天用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_par_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_fal;
}

recv_buf = emalloc(len + 1);
memt(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;
} el {
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_so黄美廉cket_error(php_sock, “unable to read from socket”, errno);
return_fal;
}

return_long(retval);
}

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

复制代码 代码如下:

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

可以看到,实际上这个函数就是调用了系统的recv而已,只是把输入参数和得到的结果都处理了一下,比较好理解。那我们再来看下socket_read,socket_read比系统的recv函数多了一个$type参数,这也是我认为这个函数存在的意义,从文档里可以看到,type有两个值,分别是php_bi世界机枪大全nary_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);

} el {

retval = recv(php_sock->bsd_socket, tmpbuf, length, 0);

}

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


复制代码 代码如下:

*t = ‘\0’;

while (*t != ‘\n’ && *t != ‘\r’ && n < maxlen) {

if (m > 0) {

t++;

n++;

} el 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 cond pass,

* and if this is nonblocking, we should return.. */

}

if (no_read > 200) {
t_errno(econnret);
return -1;
}
}

if (n < maxlen) {
m = recv(sock->bsd_socket, (void *) t, 1, flags);
}

洛阳师范 if (errno != 0 && errno != espipe && errno != eagain) {
return -1;
}

t_errno(0);
}

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

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

本文发布于:2023-04-06 09:18:31,感谢您对本站的认可!

本文链接:https://www.wtabcd.cn/fanwen/zuowen/b1a0a67986870cb2dee21661e6044fc2.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

本文word下载地址:PHP中的socket.doc

本文 PDF 下载地址:PHP中的socket.pdf

标签:代码   可以看到   函数   这两个
相关文章
留言与评论(共有 0 条评论)
   
验证码:
Copyright ©2019-2022 Comsenz Inc.Powered by © 专利检索| 网站地图