php socket fread,linux下使用fread读socket套接字的注意点

linux总所周知,一切皆文件。因此我们在读写文件时觉得fread,recv这些可以随便用。下面看看有哪些注意事项呢。

因为C库的文件处理函数较多,处理起来很灵活。所以在处理套接字时可以用fdopen函数把linux下的文件描述符转为一个C库的FILE文件指针来进行读写操作。

fread的返回值

我们知道在recv出错时会返回-1,但fread的返回值则不是这样,它不返回-1。fread/fwrite返回读到的字节数,返回的字节数小于希望读的字节数,则表明发生了错误或者读到了文件尾。要使用ferror()和feof来判断究竟发生了什么。

那么对于一个读的接口应该按下面这个模板来写,以处理文件读写发生的异常情况int read_data(FILE *fp, int len)

{    int total_len = len;    int read_len = 0;    int rtn_len = 0;    char buffer[RECV_BUF] = {0};    while(total_len)

{

read_len = MIN(total_len, RECV_BUF);

rtn_len = fread(buffer, sizeof(char), read_len, fp);        if(rtn_len 

{            if(ferror(fp))

{                if(errno == EINTR) /* 信号使读操作中断 */

{                    /* 不做处理继续往下走 */;

}                else if(errno == EAGAIN || errno == EWOULDBLOCK) /* 发生了超时 */

{

lprintf(MSG_ERROR, "socket recvice timeout: %dms\n", RCV_SND_TIMEOUT);

total_len -= rtn_len;

lprintf(MSG_DEBUG, "read len: %d\n", rtn_len);                    break;

}                else    /* 其他错误 */

{

lprintf(MSG_ERROR, "fread error: %m\n");                    break;

}

}            else    /* 读到文件尾 */

{

lprintf(MSG_ERROR, "socket closed by peer\n");

total_len -= rtn_len;

lprintf(MSG_DEBUG, "read len: %d\n", rtn_len);                break;

}

}        // lprintf(MSG_DEBUG, " %s\n", buffer);

total_len -= rtn_len;

lprintf(MSG_DEBUG, "read len: %d\n", rtn_len);

}    if(total_len != 0)

{

lprintf(MSG_ERROR, "we need to read %d bytes, but read %d bytes now\n",

len, len - total_len);        return -1;

}

}

fgets可以和recv混用吗

从目前的测试来看不可以!例如我在处理http通信时,我想用fgets来解析http的头,而后面的数据部分则使用recv来读,测试发现recv收到的数据前面少了一截。这该如何解释呢?fgets在读数据时为提高性能,从socket的接收缓冲区一次性读了很多数据到自己的缓冲区,然后返回给用户一行数据,下一次读数据则优先从C库自己的缓冲区拿数据。但是下一次你用recv来读数据了,因此它从socket的接收缓冲区读到的数据就少了一截。

fread读数据时超时的表现是什么

fread本身不支持超时设置,只可以设置阻塞非阻塞。但fread的FILE指针是通过socket转过来,而socket是可以设置接收发送超时的,所以使用fread接收socket数据时也就具有超时的属性。但表现和recv超时不太一样。

例如设置10s超时,我想recv 1000个字节,10s超时后recv会立马返回-1.而我想fread 1000个字节,你可能会看见在fread这儿不止卡10s,因为fread会尽量读满1000个字节再返回。在C库实现中,如果10s内,能收到数据,就读到自己缓冲区,然后接着收数据,直到读满1000个字节再返回。如果在10s内读不到数据那么就返回实际读到的数据,并把错误代码设置为EAGAIN。

所以假如设置10s超时,用fread读8个字节,而对端每9s发送一个字节,那么fread将卡72s然后成功返回。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值