java.sql.SQLException: 关闭的连接

本文介绍了一个Java应用程序在不同机器上运行时遇到的“关闭的连接”异常问题,并详细解释了如何通过正确判断数据库连接状态避免该问题。文章还讨论了在处理大量数据时保持连接以提高性能的方法。

最近做一个统计数据的小工具,发现在自己机器上运行正常,移到其他机器上就报:

java.sql.SQLException: 关闭的连接

的错误,开始以为是防火墙的问题,经过测试,发现和防火墙无关。

原来还是程序写得有问题,在获得数据库连接时,若重复使用Connection变量,则除了判断为空外,还要判断是否关闭,见下面代码黑体字部分。

	/**
	 * 获得数据库连接
	 * @return
	 */
	public Connection getConnection(){
		try {
			//判断有问题			
			//if(conn != null)
			//应该判断连接是否已关闭
			if(conn != null && !conn.isClosed())
     				return conn;

			String dbDriver = DBDRIVER;
			String dbUrl = DBURL;
			String dbUser = DBUSER;
			String dbPwd = DBPWD;
			
			Class.forName(dbDriver).newInstance();
			conn = DriverManager.getConnection(dbUrl, dbUser, dbPwd);
			
			if(conn == null)
				throw new Exception("获取数据库连接失败!");
		} catch (Exception e) {
			e.printStackTrace();
		}
		
		return conn;
	}



否则有可能Connection 变量不为空(null),但出现连接已关闭的问题。

查询数据表处理见下面的函数:

public void statOne(CaclCell cc) {
		try {		
			getConnection();			
			Statement st = conn.createStatement();
			String statYear = cc.getStatYear();
			String lastStatYear = cc.getLastStatYear();			
			String startDate = cc.getStartDate();
			String endDate = cc.getEndDate();
			
			String sql = cc.getSql();
			
			if(sql != null && !"".equals(sql.trim())){				
				String sql2 = sql.replace("[startDate]", startDate);
				sql2 = sql2.replace("[endDate]", endDate);
				sql2 = sql2.replace("[statYear]", statYear);
				sql2 = sql2.replace("[lastStatYear]", lastStatYear);				
				
				log.info("sql=" + sql2);
				
				ResultSet rs=st.executeQuery(sql2);
		
				rs.next();
				String result = rs.getString(1);
				cc.setResult(result);
				hmOutput.put(cc.getDataCell(), cc);
			}			conn.close();
			
		} catch (Exception e) {
			log.error("statOne failed.", e);
			try {
				if(conn != null)
					conn.close();
			} catch (Exception e2) {
				log.error("close database connection failed.", e2);
			}
		}finally{
			try {
				if(conn != null)
					conn.close();
			} catch (Exception e2) {
				log.error("close database connection failed.", e2);
			}
		}
	}


但若将代码做一些调整,错误又不出现了。

	/**
	 * 单子段统计
	 * 
	 */
	public void statOne(CaclCell cc) {
		try {	
	
			String statYear = cc.getStatYear();
			String lastStatYear = cc.getLastStatYear();			
			String startDate = cc.getStartDate();
			String endDate = cc.getEndDate();
			
			String sql = cc.getSql();
			
			if(sql != null && !"".equals(sql.trim())){				
				String sql2 = sql.replace("[startDate]", startDate);
				sql2 = sql2.replace("[endDate]", endDate);
				sql2 = sql2.replace("[statYear]", statYear);
				sql2 = sql2.replace("[lastStatYear]", lastStatYear);				
				
				log.info("sql=" + sql2);
				getConnection();				
				Statement st = conn.createStatement();
				ResultSet rs=st.executeQuery(sql2);
		
				rs.next();
				String result = rs.getString(1);
				cc.setResult(result);
				hmOutput.put(cc.getDataCell(), cc);
				
				conn.close();			
			}
						
		} catch (Exception e) {
			log.error("statOne failed.", e);
			try {
				if(conn != null)
					conn.close();
			} catch (Exception e2) {
				log.error("close database connection failed.", e2);
			}
		}
	}


也就是getConnection()紧挨executeQuery() 语句。

原因是什么呢,其实和黑体代码位置没有关系。而是如果放在if语句中,则只需要一个连接(只有一条满足要处理条件),而如果statOne()调用两次,则出现了同样的问题。

说明什么呢:conn.close(); 数据库连接关闭后,conn并不会为空。

所以要使用conn.isClosed()来判断数据库连接是否已关闭,如关闭则重新建立连接,如没关闭则继续使用;

黑体字没有正常显示,见<strong></strong>包含部分

补充说明:

连接建立需要较长时间,我的双CPU性能很好的机器要5秒左右,如果每次处理都关闭后再连接,系统运行势必会非常的慢。可以在处理完所有的数据后再关闭数据库连接,这样就能获得较高的处理性能。

### 问题分析 `java.sql.SQLException: connection closed` 错误通常表明数据库连接在尝试执行操作时已经关闭。这种问题可能由多种原因引起,例如数据库连接池配置不当、网络问题、数据库超时设置不合理等。以下是详细的解决方案。 --- ### 数据库连接池配置检查 确保使用的数据库连接池(如 Druid、HikariCP 或 C3P0)配置正确。以下是一些常见的配置参数和建议: #### Druid 连接池配置 ```properties spring.datasource.druid.initial-size=5 spring.datasource.druid.min-idle=5 spring.datasource.druid.max-active=20 spring.datasource.druid.max-wait=60000 spring.datasource.druid.time-between-eviction-runs-millis=60000 spring.datasource.druid.min-evictable-idle-time-millis=300000 spring.datasource.druid.validation-query=SELECT 1 spring.datasource.druid.test-while-idle=true spring.datasource.druid.test-on-borrow=false ``` 上述配置确保了连接池的最小空闲连接数、最大活跃连接数以及连接的有效性检测[^1]。 --- ### 数据库连接超时设置 如果数据库服务器设置了较短的连接超时时间,可能导致连接关闭。可以通过以下方式解决: #### MySQL 配置 在 MySQL 的 `my.cnf` 文件中设置以下参数: ```ini [mysqld] wait_timeout=28800 interactive_timeout=28800 ``` 将 `wait_timeout` 和 `interactive_timeout` 设置为较大的值(例如 28800 秒,即 8 小时),以避免连接过早关闭[^2]。 --- ### Spring Boot 数据源配置 在 Spring Boot 应用中,确保数据源配置正确,并启用连接有效性检测。例如: #### application.properties 配置 ```properties spring.datasource.url=jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=utf8&serverTimezone=UTC&autoReconnect=true spring.datasource.username=root spring.datasource.password=root spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver spring.datasource.type=com.alibaba.druid.pool.DruidDataSource # 连接池配置 spring.datasource.druid.test-on-borrow=true spring.datasource.druid.validation-query=SELECT 1 ``` 通过 `test-on-borrow=true` 和 `validation-query=SELECT 1` 确保每次从连接池获取连接时都会验证其有效性[^3]。 --- ### 捕获异常并重试 在代码层面,可以捕获 `SQLException` 并尝试重新执行操作。以下是一个示例: ```java import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.SQLException; public class DatabaseUtils { public static void executeQueryWithRetry(Connection connection, String query, int retryCount) throws SQLException { int attempt = 0; while (attempt < retryCount) { try (PreparedStatement statement = connection.prepareStatement(query)) { statement.executeUpdate(); return; // 如果成功执行,则退出 } catch (SQLException e) { if (e.getMessage().contains("connection closed") && attempt < retryCount - 1) { attempt++; System.out.println("Connection closed. Retrying... Attempt: " + attempt); continue; // 重试 } else { throw e; // 其他异常直接抛出 } } } } } ``` 上述代码会在捕获到 `connection closed` 异常时尝试重新执行查询[^4]。 --- ### 日志排查 启用详细的日志记录,帮助定位问题。可以在 `application.properties` 中添加以下配置: ```properties logging.level.com.alibaba.druid=DEBUG logging.level.org.springframework.jdbc=DEBUG ``` 通过查看日志,可以了解连接关闭的具体原因。 --- ### 总结 `java.sql.SQLException: connection closed` 的常见解决方案包括检查数据库连接池配置、调整数据库超时设置、启用连接有效性检测以及在代码中实现异常重试机制。根据具体场景选择合适的解决方案。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值