几个常见的 Socket 连接错误及原因

转自 http://blog.csdn.net/jifengszf/article/details/5928884  


下面列出了几个在客户与服务进程连接中常见的几个 Socket 错误,并分析了原因。后续再逐渐补充吧。

 

ECONNABORTED

          该错误被描述为“software caused connection abort”,即“软件引起的连接中止”。原因在于当服务和客户进程在完成用于 TCP 连接的“三次握手”后,客户 TCP 却发送了一个 RST (复位)分节,在服务进程看来,就在该连接已由 TCP 排队,等着服务进程调用 accept 的时候 RST 却到达了。POSIX 规定此时的 errno 值必须 ECONNABORTED。源自 Berkeley 的实现完全在内核中处理中止的连接,服务进程将永远不知道该中止的发生。服务器进程一般可以忽略该错误,直接再次调用accept。

 

accept(2) man page 写道
[ECONNABORTED] A connection arrived, but it was closed while waiting on the listen queue.

 

ECONNRESET

          该错误被描述为“connection reset by peer”,即“对方复位连接”,这种情况一般发生在服务进程较客户进程提前终止。当服务进程终止时会向客户 TCP 发送 FIN 分节,客户 TCP 回应 ACK,服务 TCP 将转入 FIN_WAIT2 状态。此时如果客户进程没有处理该 FIN (如阻塞在其它调用上而没有关闭 Socket 时),则客户 TCP 将处于 CLOSE_WAIT 状态。当客户进程再次向 FIN_WAIT2 状态的服务 TCP 发送数据时,则服务 TCP 将立刻响应 RST。一般来说,这种情况还可以会引发另外的应用程序异常,客户进程在发送完数据后,往往会等待从网络IO接收数据,很典型的如 read 或 readline 调用,此时由于执行时序的原因,如果该调用发生在 RST 分节收到前执行的话,那么结果是客户进程会得到一个非预期的 EOF 错误。此时一般会输出“server terminated prematurely”-“服务器过早终止”错误。

 

EPIPE

          错误被描述为“broken pipe”,即“管道破裂”,这种情况一般发生在客户进程不理会(或未及时处理)Socket 错误,继续向服务 TCP 写入更多数据时,内核将向客户进程发送 SIGPIPE 信号,该信号默认会使进程终止(此时该前台进程未进行 core dump)。结合上边的 ECONNRESET 错误可知,向一个 FIN_WAIT2 状态的服务 TCP(已 ACK 响应 FIN 分节)写入数据不成问题,但是写一个已接收了 RST 的 Socket 则是一个错误。

 

ETIMEDOUT

          错误被描述为“connect time out”,即“连接超时”,这种情况一般发生在服务器主机崩溃。此时客户 TCP 将在一定时间内(依具体实现)持续重发数据分节,试图从服务 TCP 获得一个 ACK 分节。当最终放弃尝试后(此时服务器未重新启动),内核将会向客户进程返回 ETIMEDOUT 错误。如果某个中间路由器判定该服务器主机已经不可达,则一般会响应“destination unreachable”-“目的地不可达”的ICMP消息,相应的客户进程返回的错误是 EHOSTUNREACH 或ENETUNREACH。当服务器重新启动后,由于 TCP 状态丢失,之前所有的连接信息也不存在了,此时对于客户端发来请求将回应 RST。如果客户进程对检测服务器主机是否崩溃很有必要,要求即使客户进程不主动发送数据也能检测出来,那么需要使用其它技术,如配置 SO_KEEPALIVE Socket 选项,或实现某些心跳函数。

 

ENOPROTOOPT

          该错误不是一个 Socket 连接相关的错误。errno 给出该值可能由于,通过 getsockopt 系统调用来获得一个套接字的当前选项状态时,如果发现了系统不支持的选项参数就会引发该错误。

 

          一定要检查 write 方法的返回值,尤其是服务端程序,当返回 -1 的时候很有可能是“connection reset by peer”(ECONNRESET 104)。如果服务程序没有处理 SIGPIPE 信号的话,第二次程序在这条已经 close 的 socket 再次 write 时 SIGPIPE 信号就发送到 socket 关联的 owen 进程,也就是上面说的管道破裂,而该信号的默认处理是结束进程。今天不小心又因为这个浪费了两小时,在客户程序连续通信的时候,直接结束客户进程就造成服 务进程也同时退出。开来还是太粗心。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 当socket断开连接时,有多个原因可能会导致这种情况发生。其中一些原因可能是网络问题、服务器故障、客户端故障或者用户主动断开连接。在程序中,我们可以通过捕获异常来处理这种情况。在Python中,可以使用try-except语句来捕获异常,在发生异常时执行一些特定的操作,比如关闭socket连接、重新连接等。以下是一个简单的示例代码: ```python import socket HOST = 'localhost' PORT = 8000 try: # 创建socket连接 s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) s.connect((HOST, PORT)) # 发送数据 s.sendall(b'Hello, world') # 接收数据 data = s.recv(1024) print('Received', repr(data)) except Exception as e: print(e) finally: # 关闭socket连接 s.close() ``` 在上面的代码中,我们使用了try-except语句来捕获异常。如果在连接、发送或接收数据的过程中发生了异常,程序会输出异常信息并关闭socket连接。如果没有发生异常,程序会正常关闭socket连接。 ### 回答2: 当Socket连接断开时,意味着建立的网络连接被中断或关闭。有几种情况会导致Socket连接断开: 1. 服务器或客户端异常关闭:如果服务器或客户端应用程序关闭或崩溃,Socket连接将会断开。这可能是由于程序中的bug、操作系统问题或网络中断引起的。 2. 网络中断:如果网络中断或出现问题,Socket连接也会断开。这可能是由于网络故障、网络设备故障或网络拥塞等原因引起的。 3. 超时:Socket连接在一定的时间内没有收到任何数据传输时,就会发生超时。超时可以是因为网络延迟、对方应用程序崩溃、服务器负载过重等原因导致的。 4. 手动关闭:应用程序可以通过调用Socket的close()方法来主动关闭连接。这通常在通信完成或不再需要连接时使用。 无论何种原因导致Socket连接断开,当连接断开时,通常会触发错误或异常,并且连接双方都会收到有关连接断开的通知。应用程序可以根据这些通知采取相应的措施,例如重新连接、重新建立资源或提示用户重新尝试连接。 ### 回答3: 当一个socket断开连接时,意味着通信的两个端点之间的连接已经被中断或关闭。这种情况可能由多种原因引起,如网络故障、通信错误连接超时或用户操作等。 一旦发生socket断开连接,通信双方将无法继续进行数据交换。数据将无法传送到远程端口,并且对方也无法发送任何数据到本地端口。这也意味着任何未收到的数据将丢失,正在发送的数据也无法到达目的地。 在程序设计中,可以通过捕获特定的断开连接错误或异常来处理socket断开连接的情况。一般来说,当socket断开连接时,可以采取以下几种操作: 1. 重新建立连接:可以尝试重新连接到远程主机,重新建立通信链路。这通常需要重新创建socket对象,并重新绑定到远程主机的IP地址和端口号。 2. 进行错误处理:如果socket断开连接是由于网络故障等原因引起的,可以根据具体错误类型来处理异常。例如,可以记录错误日志、向用户显示错误信息或提示用户重新连接等。 3. 清理资源:当socket断开连接时,可能需要释放相关的资源,如关闭socket对象、释放内存或清理缓冲区。这可以防止资源泄露和占用。 总之,socket断开连接是在网络通信过程中常见的情况,需要注意处理,并根据具体情况采取相应的措施来处理错误和恢复通信。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值