SocketTimeoutException: Read timed out

本文详细分析了在调用第三方接口时遇到的SocketTimeoutException,探讨了其原因,如网络问题和服务端响应超时,并提供了通过调整qps和使用连接池来避免异常的策略。

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

一,问题

调用第三方接口,报错,如下

org.springframework.web.client.ResourceAccessException: I/O error on POST request for "https://a1-yl365.saicmaxus.com/xdap-open/open/process/v1/submit": Read timed out; nested exception is java.net.SocketTimeoutException: Read timed out

二,原因

客户端读取服务端数据超时,-》 服务端在一定时间内没有响应客户端,-〉

原因:网络问题/服务端响应超时

三,解决方案

实际场景中,需要调用第三方写入的数据量为2200左右,已经对qps进行了限制,每当请求达到20,就休眠1s,控制qps20

改进:降低qps为5

代码如下

    private void insertAppass(List<VehicleAnnualWoSaveProReqDTO> saveDTOS) {
        int count = 0;
        for (VehicleAnnualWoSaveProReqDTO data : saveDTOS) {
            count++;
            if (count >= insertLimit) {
                try {
                    count = 0;
                    Thread.sleep(1000L);
                } catch (InterruptedException e) {
                    log.error("年检工单 执行异常 线程中断", e);
                    Thread.currentThread().interrupt();
                }
            }
            // 异步执行任务
            CompletableFuture<Void> runFuture = CompletableFuture.runAsync(() -> xdapCommonService.xdapSingleInsertV2(JSON.toJSONString(data)));
            runFuture.exceptionally(e -> {
                log.error("年检工单 执行异常,调度得帆api保存工单信息,工单信息: {}", JSON.toJSONString(data), e);
                return null;
            });
        }
    }

四,参考

java.net.SocketTimeoutException: Read timed out_read timed out; nested exception is java.net.socke-CSDN博客

引言
在进行网络编程时,我们经常会遇到java.net.SocketTimeoutException: Read timed out异常,这个异常通常在网络通信过程中出现,给开发者带来了一定的困惑。本文将深入解析SocketTimeoutException异常的原因,并提供一些避免该异常的策略。

什么是SocketTimeoutException异常?
SocketTimeoutException异常是java.net.SocketTimeoutException的一种异常情况。它通常在进行网络通信时出现,当一个读操作在指定的时间内没有完成时,Java网络编程会抛出SocketTimeoutException异常。

以下是一个示例代码,展示了这个异常的触发情况:

public void performNetworkOperation() {
    try {
        // 创建Socket对象
        Socket socket = new Socket("example.com", 80);
        
        // 设置读取超时时间
        socket.setSoTimeout(5000);
        
        // 执行网络操作
        // ...
    } catch (SocketTimeoutException e) {
        // 处理SocketTimeoutException异常
        // ...
    } catch (IOException e) {
        // 处理其他IO异常
        // ...
    }
}
在上述代码中,我们创建了一个Socket对象,并通过setSoTimeout()方法设置读取超时时间为5秒。如果在5秒内没有完成读取操作,Java网络编程会抛出SocketTimeoutException异常。

异常产生的原因
SocketTimeoutException异常的产生是由于网络通信过程中读取操作未能在指定的超时时间内完成。这可能是因为网络延迟、服务器响应时间过长或者网络连接不稳定等原因。

解析SocketTimeoutException异常
要深入理解SocketTimeoutException异常,我们需要了解其继承关系。SocketTimeoutException是java.io.InterruptedIOException的子类,而java.io.InterruptedIOException又是java.io.IOException的子类。因此,SocketTimeoutException异常是IOException的一种特殊情况。

SocketTimeoutException异常提供了一个重要的方法getTimeout(),该方法返回引发异常的超时时间。通过调用getTimeout()方法,我们可以获取到超时时间,并根据需要进行相应的处理。

以下是一个示例代码,展示了如何使用getTimeout()方法来处理超时时间:

public void performNetworkOperation() {
    try {
        // 创建Socket对象
        Socket socket = new Socket("example.com", 80);
        
        // 设置读取超时时间
        int timeout = 5000;
        socket.setSoTimeout(timeout);
        
        // 执行网络操作
        // ...
    } catch (SocketTimeoutException e) {
        // 获取超时时间
        int timeout = e.getTimeout();
        
        // 处理SocketTimeoutException异常
        // ...
    } catch (IOException e) {
        // 处理其他IO异常
        // ...
    }
}

通过使用getTimeout()方法,我们可以获取引发异常的超时时间,并根据具体的业务需求进行相应的处理。

避免SocketTimeoutException异常的策略
为了避免SocketTimeoutException异常的发生,我们可以采取以下策略:

1. 设置合理的超时时间
在进行网络通信时,我们应该根据实际情况设置合理的超时时间。如果超时时间设置得过短,可能会导致读取操作在没有完成之前就抛出SocketTimeoutException异常;而如果超时时间设置得过长,可能会导致应用程序在网络故障的情况下长时间等待。

public void performNetworkOperation() {
    try {
        // 创建Socket对象
        Socket socket = new Socket("example.com", 80);
        
        // 设置合理的读取超时时间
        int timeout = 5000;
        socket.setSoTimeout(timeout);
        
        // 执行网络操作
        // ...
    } catch (SocketTimeoutException e.getTimeout()

方法来获取引发异常的超时时间,并根据需要进行相应的处理。

2. 检查网络连接状态
在进行网络通信之前,我们应该先检查网络连接状态,确保网络连接正常。可以使用Java的网络工具类如InetAddress或URL来检查网络连接状态。

以下是一个示例代码,展示了如何检查网络连接状态:

public boolean isNetworkConnected() {
    try {
        // 检查连接到一个可靠的IP地址
        InetAddress.getByName("example.com");
        return true;
    } catch (UnknownHostException e) {
        return false;
    }
}

在进行网络通信之前,我们可以调用isNetworkConnected()方法来检查网络连接状态。如果返回true,则表示网络连接正常,可以进行网络通信;如果返回false,则表示网络连接异常,可以选择进行相应的处理,例如提示用户检查网络连接或进行重试操作。

3. 使用连接池
在高并发的网络环境下,频繁地创建和销毁Socket连接可能会增加系统的负担并导致SocketTimeoutException异常的发生。为了避免这种情况,我们可以使用连接池来管理和复用Socket连接。

连接池可以在系统启动时创建一定数量的Socket连接,并将这些连接存储在连接池中。当需要进行网络通信时,可以从连接池中获取一个可用的连接,使用完后再将连接放回连接池中供其他线程使用。

使用连接池可以减少Socket连接的创建和销毁次数,提高系统的性能和稳定性。

结论
在本文中,我们深入解析了SocketTimeoutException异常的原因,并提供了一些避免该异常的策略。通过设置合理的超时时间、检查网络连接状态和使用连接池等方法,我们可以有效地避免SocketTimeoutException异常的发生,提高网络编程的稳定性和可靠性。
————————————————

                            版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
                        
原文链接:https://blog.csdn.net/weixin_42373241/article/details/135956097

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值