SQL 悬空连接:我在哪里没有正确关闭连接?
SQL connections dangling: Where am I not correctly closing up connections correctly?
我正在构建一个基本的 java 应用程序以将一些文件加载到 mysql 数据库中。我能够毫无问题地加载文件并填充我的 tables。然而,在与审查我的代码的人交谈后,我显然没有正确关闭我的连接并浪费资源。我在哪里不关闭连接?我做错了吗?
我在我的 DbSinger class 中使用 try-with-resources 构造来对我的数据库执行准备好的语句,只要满足以下条件,它就会自动关闭连接实现了 AutoCloseable 接口,它位于 Db 的父级 class 中。然而,close() 方法从未达到。 DbSinger 在我的 main() 中实例化,然后 运行s 它是单一方法 populateSingers() 和 Singer 个对象的 ArrayList。
连接Class
public class SQLConnection {
private static final String servername = "localhost";
private static final int port = 3306;
private static final String user = "ng_user";
private static final String pass = "ng";
private static final String db = "ng_music";
private static final String connectionString = "jdbc:mysql://" + servername + ":" + port + "/" + db;
public Connection provide() {
try {
Class.forName("com.mysql.cj.jdbc.Driver");
return DriverManager.getConnection(connectionString, user, pass);
}
catch (SQLException | ClassNotFoundException e) {
throw new SQLConnectionException(e);
}
}
public class SQLConnectionException extends RuntimeException {
SQLConnectionException(Exception e) {super(e);}
}
}
抽象父级 class
public abstract class Db implements AutoCloseable{
private Connection connection;
Db() {
SQLConnection sqlC = new SQLConnection();
this.connection = sqlC.provide();
}
@Override
public synchronized void close() throws SQLException {
if(connection != null) {
connection.close();
connection = null;
System.out.println("Connection closed");
}
}
Connection getConnection() {
return connection;
}
boolean checkIfPopulated(String query){
try {
PreparedStatement ps = getConnection().prepareStatement(query);
ResultSet rs = ps.executeQuery();
return !rs.next();
} catch (SQLException e) {
e.printStackTrace();
}
return true;
}
}
具体 class 执行歌手数据库查询 table
public class DbSinger extends Db {
public DbSinger() {
super();
}
public void populateSingers(ArrayList<Singer> singers) {
String populateSingersQuery = "insert into ng_singers(name, dob, sex) values(?,?,?)";
if(!checkIfPopulated("select * from ng_singers")){
System.out.println("Singer Table is already populated");
return;
}
try (PreparedStatement ps = getConnection().prepareStatement(populateSingersQuery)) {
for (Singer s : singers) {
ps.setString(1, s.getName());
ps.setDate(2, java.sql.Date.valueOf(s.getDob()));
ps.setString(3, s.getSex());
ps.addBatch();
}
ps.executeBatch();
System.out.println("Singers added to table");
} catch (SQLException e) {
e.printStackTrace();
}
}
}
我的代码可以 运行 正常执行并执行所需的操作,但我想了解我不关闭连接的原因和位置,并了解如何解决此问题。或者至少明白我是不是处理错了。
您的代码是旧方法。而且您确实需要手动关闭。但是,对于 Java 8,您可以像下面那样使用 try with resource
,
try (Connection conn = ds.getConnection();
Statement stmt = conn.createStatement()) {
try {
stmt.execute(dropsql);
} catch (Exception ignore) {} // ignore if table not dropped
stmt.execute(createsql);
stmt.execute(insertsql);
try (ResultSet rs = stmt.executeQuery(selectsql)) {
rs.next();
} catch (Exception e2) {
e2.printStackTrace();
return("failed");
}
} catch(Exception e) {
e.printStackTrace();
return("failed");
}
在您的情况下,您需要在 try-with-resources
语句中实例化 DBSinger class 以关闭基础连接。
而不是做:
DbSinger dbSinger = new DbSinger();
您需要做的:
try (DbSinger dbSinger = new DbSinger()) {
// Your other code
}
这样,您在 Db
class 中覆盖的 close()
方法将被自动调用。
此外,通过以下方式关闭您在 checkIfPopulated
方法中创建的 preparedStatement
:
try (PreparedStatement ps = getConnection().prepareStatement(query)) {
// Other codes
}
我正在构建一个基本的 java 应用程序以将一些文件加载到 mysql 数据库中。我能够毫无问题地加载文件并填充我的 tables。然而,在与审查我的代码的人交谈后,我显然没有正确关闭我的连接并浪费资源。我在哪里不关闭连接?我做错了吗?
我在我的 DbSinger class 中使用 try-with-resources 构造来对我的数据库执行准备好的语句,只要满足以下条件,它就会自动关闭连接实现了 AutoCloseable 接口,它位于 Db 的父级 class 中。然而,close() 方法从未达到。 DbSinger 在我的 main() 中实例化,然后 运行s 它是单一方法 populateSingers() 和 Singer 个对象的 ArrayList。
连接Class
public class SQLConnection {
private static final String servername = "localhost";
private static final int port = 3306;
private static final String user = "ng_user";
private static final String pass = "ng";
private static final String db = "ng_music";
private static final String connectionString = "jdbc:mysql://" + servername + ":" + port + "/" + db;
public Connection provide() {
try {
Class.forName("com.mysql.cj.jdbc.Driver");
return DriverManager.getConnection(connectionString, user, pass);
}
catch (SQLException | ClassNotFoundException e) {
throw new SQLConnectionException(e);
}
}
public class SQLConnectionException extends RuntimeException {
SQLConnectionException(Exception e) {super(e);}
}
}
抽象父级 class
public abstract class Db implements AutoCloseable{
private Connection connection;
Db() {
SQLConnection sqlC = new SQLConnection();
this.connection = sqlC.provide();
}
@Override
public synchronized void close() throws SQLException {
if(connection != null) {
connection.close();
connection = null;
System.out.println("Connection closed");
}
}
Connection getConnection() {
return connection;
}
boolean checkIfPopulated(String query){
try {
PreparedStatement ps = getConnection().prepareStatement(query);
ResultSet rs = ps.executeQuery();
return !rs.next();
} catch (SQLException e) {
e.printStackTrace();
}
return true;
}
}
具体 class 执行歌手数据库查询 table
public class DbSinger extends Db {
public DbSinger() {
super();
}
public void populateSingers(ArrayList<Singer> singers) {
String populateSingersQuery = "insert into ng_singers(name, dob, sex) values(?,?,?)";
if(!checkIfPopulated("select * from ng_singers")){
System.out.println("Singer Table is already populated");
return;
}
try (PreparedStatement ps = getConnection().prepareStatement(populateSingersQuery)) {
for (Singer s : singers) {
ps.setString(1, s.getName());
ps.setDate(2, java.sql.Date.valueOf(s.getDob()));
ps.setString(3, s.getSex());
ps.addBatch();
}
ps.executeBatch();
System.out.println("Singers added to table");
} catch (SQLException e) {
e.printStackTrace();
}
}
}
我的代码可以 运行 正常执行并执行所需的操作,但我想了解我不关闭连接的原因和位置,并了解如何解决此问题。或者至少明白我是不是处理错了。
您的代码是旧方法。而且您确实需要手动关闭。但是,对于 Java 8,您可以像下面那样使用 try with resource
,
try (Connection conn = ds.getConnection();
Statement stmt = conn.createStatement()) {
try {
stmt.execute(dropsql);
} catch (Exception ignore) {} // ignore if table not dropped
stmt.execute(createsql);
stmt.execute(insertsql);
try (ResultSet rs = stmt.executeQuery(selectsql)) {
rs.next();
} catch (Exception e2) {
e2.printStackTrace();
return("failed");
}
} catch(Exception e) {
e.printStackTrace();
return("failed");
}
在您的情况下,您需要在 try-with-resources
语句中实例化 DBSinger class 以关闭基础连接。
而不是做:
DbSinger dbSinger = new DbSinger();
您需要做的:
try (DbSinger dbSinger = new DbSinger()) {
// Your other code
}
这样,您在 Db
class 中覆盖的 close()
方法将被自动调用。
此外,通过以下方式关闭您在 checkIfPopulated
方法中创建的 preparedStatement
:
try (PreparedStatement ps = getConnection().prepareStatement(query)) {
// Other codes
}