socket或pipe 一端断开; errno 104:connetction reset by peer的错误分析

本文深入解析了errno=104错误,即connection reset by peer的现象,常见于TCP通信中,当一端提前关闭连接而另一端仍尝试写入数据时触发。文章探讨了错误产生的根本原因,包括数据长度不匹配的情况,并详细解释了TCP四次握手过程中的FIN与RST包的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

errno 104:connetction reset by peer的错误分析

https://blog.youkuaiyun.com/alibo2008/article/details/45694845

errno = 104错误表明你在对一个对端socket已经关闭的的连接调用write或send方法,在这种情况下,调用write或send方法后,对端socket便会向本端socket发送一个RESET信号,在此之后如果继续执行write或send操作,就会得到errno为104,错误描述为connection reset by peer。

        出现这种问题的很大一部分原因,至少我遇到的几次全都是,发送端和接收端事先约定好的数据长度不一致导致的,接收端被通知要收的数据长度小于发送端实际要发送的数据长度。比如接收端被通知要收1024个字节,但发送端却发了1025(可以是字符串末尾隐含的结束符),这样一来,接收端收完1024就执行了close操作,如果发送端再继续发送,接收端协议就会向发送端返回一个RESET信号,RESET信号可以抓包看到,如下所示:

  具体的分析可以结合TCP的"四次握手"关闭. TCP是全双工的信道, 可以看作两条单工信道, TCP连接两端的两个端点各负责一条. 当对端调用close时, 虽然本意是关闭整个两条信道, 但本端只是收到FIN包. 按照TCP协议的语义, 表示对端只是关闭了其所负责的那一条单工信道, 仍然可以继续接收数据. 也就是说, 因为TCP协议的限制, 一个端点无法获知对端的socket是调用了close还是shutdown.

对于一个TCP连接,如果对端执行close操作,则会向本端发送一个FIN分节,这时候读本端socket会返回0,我们就知道对方已经关闭了连接,通常这时候我们会在本地调用close来主动关闭本端连接。但如果对方socket已经执行了close的操作,本端socket还继续在这个连接上写数据,就会触发对端socket发送RST报文,按照TCP的四次握手原理,这时候本端socket应该也要开始执行close的操作流程了,而不是接着发数据.


  •                     <li class="tool-item tool-active is-like "><a href="javascript:;"><svg class="icon" aria-hidden="true">
                            <use xlink:href="#csdnc-thumbsup"></use>
                        </svg><span class="name">点赞</span>
                        <span class="count">1</span>
                        </a></li>
                        <li class="tool-item tool-active is-collection "><a href="javascript:;" data-report-click="{&quot;mod&quot;:&quot;popu_824&quot;}"><svg class="icon" aria-hidden="true">
                            <use xlink:href="#icon-csdnc-Collection-G"></use>
                        </svg><span class="name">收藏</span></a></li>
                        <li class="tool-item tool-active is-share"><a href="javascript:;"><svg class="icon" aria-hidden="true">
                            <use xlink:href="#icon-csdnc-fenxiang"></use>
                        </svg>分享</a></li>
                        <!--打赏开始-->
                                                <!--打赏结束-->
                                                <li class="tool-item tool-more">
                            <a>
                            <svg t="1575545411852" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="5717" xmlns:xlink="http://www.w3.org/1999/xlink" width="200" height="200"><defs><style type="text/css"></style></defs><path d="M179.176 499.222m-113.245 0a113.245 113.245 0 1 0 226.49 0 113.245 113.245 0 1 0-226.49 0Z" p-id="5718"></path><path d="M509.684 499.222m-113.245 0a113.245 113.245 0 1 0 226.49 0 113.245 113.245 0 1 0-226.49 0Z" p-id="5719"></path><path d="M846.175 499.222m-113.245 0a113.245 113.245 0 1 0 226.49 0 113.245 113.245 0 1 0-226.49 0Z" p-id="5720"></path></svg>
                            </a>
                            <ul class="more-box">
                                <li class="item"><a class="article-report">文章举报</a></li>
                            </ul>
                        </li>
                                            </ul>
                </div>
                                <div class="right-toolbox"><a href="https://blog.youkuaiyun.com/zjk2752/article/details/21236725" target="_blank" class="jump-net-article">
                <svg t="1575545252354" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="5597" xmlns:xlink="http://www.w3.org/1999/xlink" width="200" height="200"><defs><style type="text/css"></style></defs><path d="M243.9 1022.2c-62.3 0-124-23.8-171.5-70.8C26.4 905.5 1.5 845 1.5 779.9s24.9-125.6 70.8-171.5l184-184.1c45.9-45.9 106.4-70.8 171.5-70.8s125.6 24.9 171.5 70.8c18.1 18.1 18.1 47 0 65.1s-47 18.1-65.1 0c-28.3-28.3-65.7-43.5-105.9-43.5s-78.1 15.3-105.9 43.7l-184 184c-58.3 58.3-58.3 153.4 0 212.3 28.3 28.3 65.7 43.5 105.9 43.5s78.1-15.3 105.9-43.5l184-184c18.1-18.1 47-18.1 65.1 0 18.1 18.1 18.1 47 0 65.1l-184 184c-46.9 48-109.1 71.2-171.4 71.2z m523.7-423l184-184c94.5-94.5 94.5-248 0-342.5s-248-94.5-342.5 0l-184 184c-18.1 18.1-18.1 47 0 65.1s47 18.1 65.1 0l184-184c28.3-28.3 65.7-43.5 105.9-43.5s78.1 15.3 105.9 43.5c58.3 58.3 58.3 153.4 0 212.3l-184 184c-58.3 58.3-153.4 58.3-212.3 0-18.1-18.1-47-18.1-65.1 0-18.1 18.1-18.1 47 0 65.1 47 47 109.3 70.8 171.5 70.8s123.9-23.2 171.5-70.8z" p-id="5598"></path></svg>
                    站内首发文章</a></div>
                            </div>
            <div class="person-messagebox">
                <div class="left-message"><a href="https://blog.youkuaiyun.com/alibo2008">
                    <img src="https://profile.csdnimg.cn/9/6/C/3_alibo2008" class="avatar_pic" username="alibo2008">
                                            <img src="https://g.csdnimg.cn/static/user-reg-year/2x/12.png" class="user-years">
                                    </a></div>
                <div class="middle-message">
                                        <div class="title"><span class="tit"><a href="https://blog.youkuaiyun.com/alibo2008" data-report-click="{&quot;mod&quot;:&quot;popu_379&quot;}" target="_blank">alibo2008</a></span>
                                            </div>
                    <div class="text"><span>发布了21 篇原创文章</span> · <span>获赞 8</span> · <span>访问量 12万+</span></div>
                </div>
                                <div class="right-message">
                                            <a href="https://im.youkuaiyun.com/im/main.html?userName=alibo2008" target="_blank" class="btn btn-sm btn-red-hollow bt-button personal-letter">私信
                        </a>
                                                            <a class="btn btn-sm  bt-button personal-watch" data-report-click="{&quot;mod&quot;:&quot;popu_379&quot;}">关注</a>
                                    </div>
                            </div>
                    </div>
    
### Socket 异常 `Connection reset by peer (104)` 的解决方案 `Connection reset by peer` 是网络通信中常见的异常,表示客户端服务端在连接过程中,对方突然关闭了连接。错误104(ECONNRESET)表明该连接已被对端强制终止。以下是针对不同场景的解决方案。 #### 1. **Apache AB 压力测试中的 Connection Reset by Peer** 当使用 Apache Benchmark (`ab`) 进行高并发压力测试时,可能会出现 `apr_socket_recv: Connection reset by peer (104)` 错误。这通常是由于服务器端无法处理大量并发请求导致的。 - **调整内核参数**:增加系统允许的最大连接数和文件描述符限制。例如,修改 `/etc/security/limits.conf` 文件以提升 `nofile` 上限: ``` * soft nofile 65536 * hard nofile 65536 ``` 同时修改 `/etc/sysctl.conf` 中的 `net.core.somaxconn` 和 `net.ipv4.tcp_max_syn_backlog` 等参数[^4]。 - **优化 Web 服务器配置**:对于 Apache Nginx,确保 `KeepAlive` 设置合理,并根据负载调整 `MaxClients`、`ThreadsPerChild` 等参数。 #### 2. **Redis 客户端连接问题** 在 Python 中使用 `redis-py` 库时,如果遇到 `redis.exceptions.ConnectionError: Error 104 while writing to socket. Connection reset by peer.`,可能是由于连接池管理不当 Redis 服务端主动断开连接。 - **设置健康检查间隔**:通过 `health_check_interval` 参数定期检测连接状态,适用于连接建立后中途断开的情况: ```python client = redis.Redis(host='localhost', port=6379, db=0, health_check_interval=30) ``` 此方法不适用于初始连接失败的情况[^2]。 #### 3. **Paramiko SSH 连接异常** 使用 Paramiko 库进行 SSH 连接时,若出现 `ConnectionResetError: [Errno 104] Connection reset by peer`,可能是由于 Paramiko 尝试使用本地 SSH Agent 导致的问题。 - **禁用 SSH Agent 支持**:在连接时设置 `allow_agent=False` 可绕过 SSH Agent 相关逻辑: ```python import paramiko ssh = paramiko.SSHClient() ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy()) ssh.connect(hostname='example.com', username='user', password='pass', allow_agent=False) ``` 此外,检查远程主机的 SSH 配置是否正常,如 `sshd_config` 是否启用密码登录公钥认证正确[^3]。 #### 4. **通用 TCP 层面调优与排查** - **检查防火墙安全策略**:某些防火墙负载均衡器可能在空闲超时后中断连接。确认网络设备云服务商的超时设置。 - **调整 TCP 参数**:通过以下内核参数优化 TCP 行为: - `net.ipv4.tcp_keepalive_time` - `net.ipv4.tcp_keepalive_intvl` - `net.ipv4.tcp_keepalive_probes` 这些参数控制 TCP Keepalive 探测的时间间隔和次数,有助于发现并关闭失效连接。 - **应用层心跳机制**:在客户端和服务端实现应用层心跳包,定期发送探测消息以维持连接活跃状态。 #### 5. **资源限制与性能瓶颈** - **检查系统资源限制**:使用 `ulimit -a` 查看当前用户打开文件数限制,若接近上限则需调整 `/etc/security/limits.conf` 并重启服务[^4]。 - **监控系统负载与内存使用**:高并发场景下,若服务器 CPU 内存资源耗尽,也可能导致连接被强制中断。使用 `top`、`htop`、`vmstat` 等工具分析系统性能瓶颈。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值