TCP/IP 错误原因码 (errno) 各种异常现象和分析

理解 TCP

https://segmentfault.com/blog/iamjerryc

TCP/IP 某些最常见的错误原因码 (errno)列表(转)

https://www.cnblogs.com/qq78292959/archive/2013/04/22/3036103.html

从TCP协议到TCP通信的各种异常现象和分析(上)

https://network.51cto.com/art/201905/596531.htm

EAGAIN、EWOULDBLOCK、EINTR与非阻塞 长连接

http://blog.chinaunix.net/uid-23849526-id-120552.html

EAGAIN、EWOULDBLOCK、EINTR与非阻塞 长连接
EWOULDBLOCK用于非阻塞模式,不需要重新读或者写

EINTR指操作被中断唤醒,需要重新读/写

在Linux环境下开发经常会碰到很多错误(设置errno),其中EAGAIN是其中比较常见的一个错误(比如用在非阻塞操作中)。
从字面上来看,是提示再试一次。这个错误经常出现在当应用程序进行一些非阻塞(non-blocking)操作(对文件或socket)的时候。例如,以 O_NONBLOCK的标志打开文件/socket/FIFO,如果你连续做read操作而没有数据可读。此时程序不会阻塞起来等待数据准备就绪返 回,read函数会返回一个错误EAGAIN,提示你的应用程序现在没有数据可读请稍后再试。
又例如,当一个系统调用(比如fork)因为没有足够的资源(比如虚拟内存)而执行失败,返回EAGAIN提示其再调用一次(也许下次就能成功)。
Linux - 非阻塞socket编程处理EAGAIN错误
在linux进行非阻塞的socket接收数据时经常出现Resource temporarily unavailable,errno代码为11(EAGAIN),这是什么意思?
这表明你在非阻塞模式下调用了阻塞操作,在该操作没有完成就返回这个错误,这个错误不会破坏socket的同步,不用管它,下次循环接着recv就可以。 对非阻塞socket而言,EAGAIN不是一种错误。在VxWorks和Windows上,EAGAIN的名字叫做EWOULDBLOCK。
另外,如果出现EINTR即errno为4,错误描述Interrupted system call,操作也应该继续。
最后,如果recv的返回值为0,那表明连接已经断开,我们的接收操作也应该结束。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
TCP/IP是一种网络通信协议,它是互联网的基础协议之一。TCP/IP协议族包括传输层的TCP协议和UDP协议,以及网络层的IP协议和ICMP协议等。 TCP/IP通信的基本流程: 1.客户端发送连接请求到服务器端; 2.服务器端接收到请求,并发送回应信息; 3.客户端接收到回应信息,建立连接; 4.客户端和服务器端进行数据传输; 5.客户端和服务器端断开连接。 下面是一个TCP/IP通信的C语言代码示例: 客户端代码: #include <stdio.h> #include <stdlib.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <errno.h> #include <string.h> int main(int argc, char **argv) { int sockfd, n; char recvline[1024], sendline[1024]; struct sockaddr_in servaddr; if (argc != 2) { printf("usage: ./client <ipaddress>\n"); exit(1); } if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) { printf("socket error: %s\n", strerror(errno)); exit(1); } bzero(&servaddr, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_port = htons(1234); if (inet_pton(AF_INET, argv[1], &servaddr.sin_addr) <= 0) { printf("inet_pton error: %s\n", strerror(errno)); exit(1); } if (connect(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0) { printf("connect error: %s\n", strerror(errno)); exit(1); } while (fgets(sendline, 1024, stdin) != NULL) { if (write(sockfd, sendline, strlen(sendline)) < 0) { printf("write error: %s\n", strerror(errno)); exit(1); } if ((n = read(sockfd, recvline, 1024)) < 0) { printf("read error: %s\n", strerror(errno)); exit(1); } recvline[n] = '\0'; if (fputs(recvline, stdout) == EOF) { printf("fputs error: %s\n", strerror(errno)); exit(1); } } if (n < 0) { printf("read error: %s\n", strerror(errno)); exit(1); } exit(0); } 服务器端代码: #include <stdio.h> #include <stdlib.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <errno.h> #include <string.h> int main(int argc, char **argv) { int listenfd, connfd, n; struct sockaddr_in servaddr, cliaddr; socklen_t len; char buf[1024]; if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) { printf("socket error: %s\n", strerror(errno)); exit(1); } bzero(&servaddr, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_addr.s_addr = htonl(INADDR_ANY); servaddr.sin_port = htons(1234); if (bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0) { printf("bind error: %s\n", strerror(errno)); exit(1); } if (listen(listenfd, 5) < 0) { printf("listen error: %s\n", strerror(errno)); exit(1); } while (1) { len = sizeof(cliaddr); if ((connfd = accept(listenfd, (struct sockaddr *)&cliaddr, &len)) < 0) { printf("accept error: %s\n", strerror(errno)); continue; } printf("connection from %s, port %d\n", inet_ntoa(cliaddr.sin_addr), ntohs(cliaddr.sin_port)); while ((n = read(connfd, buf, 1024)) > 0) { buf[n] = '\0'; printf("recv: %s", buf); if (write(connfd, buf, n) < 0) { printf("write error: %s\n", strerror(errno)); exit(1); } } if (n < 0) { printf("read error: %s\n", strerror(errno)); exit(1); } close(connfd); } exit(0); } 以上代码实现了一个简单的TCP/IP通信模型,客户端通过输入信息发送到服务器端,服务器端接收到信息后将信息原样返回给客户端。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值