如何在以下几种方法中关闭 resultSet、prepareStatement、conn 以避免 rs 关闭和连接池被卡住

Posted

技术标签:

【中文标题】如何在以下几种方法中关闭 resultSet、prepareStatement、conn 以避免 rs 关闭和连接池被卡住【英文标题】:how can i close the resultSet, prepareStatement, conn in several methods below to avoid rs close and connection pool getting jammed 【发布时间】:2012-01-27 14:01:54 【问题描述】:

整个数据操作如下。我想在不干扰下一个连接的情况下关闭每个资源。我应该将构造函数更改为 connection() 方法然后有一个 disconnect() 方法,但是这样做之后我应该在哪里

public class DataBean


    private Connection conn = null;
    private ResultSet res = null;
    private InitialContext context;
    private DataSource datasource;
    private Statement stmt=null;
    private java.sql.PreparedStatement prepar = null;
    private java.sql.CallableStatement proc = null;
    public static int PAGECOUNT; //²éѯºó·µ»ØµÄ×ÜÒ³Êý ÒòΪjavaµÄº¯Êý²»ÄÜ´«ÒýÓÃËùÒÔÐèÒªÓþ²Ì¬±äÁ¿À´»ñµÃ
    public DataBean()
    
        try 

            Class.forName("com.mysql.jdbc.Driver");
            conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/MyData", "root","loikenu");
            context = new InitialContext();
            datasource = (DataSource)context.lookup("jdbc/MyData");
            conn = datasource.getConnection();


            //stmt =conn.createStatement();
         catch (Exception e) 
            JOptionPane.showMessageDialog(null, e.getMessage());
        
    


    public UserBean checkUsersLogin(String userName, String userPwd) //µÇ½ÑéÖ¤
    
        UserBean ub = null;
        if (!checkParameter(userName + userPwd))
        
            userName = "null";
            userPwd = "null";
        
        try
        
            String sql =
                "select count(*) from admin where userName=? and userPwd=?";

            prepar = conn.prepareStatement(sql);
            //set parameter values for preparedstatment object
            prepar.setString(1, userName);
            prepar.setString(2, userPwd);
            //execute query using preparedstatement object
            res = prepar.executeQuery();
            if (res.next())
            
                //get data from reults set returned by jdbc
                if (res.getInt(1) > 0)
                
                    ub = this.getUser(userName);
                
                else
                
                    ub = null;
                
            
        
        catch (Exception e)
        
            ub = null;
            e.printStackTrace();
        
        return ub;
    

    public UserBean getUser(String userName) //ÌáÈ¡µÇ½Óû§ÐÅÏ¢
    
        UserBean ub = new UserBean();
        int i=1;
        String sql = "select * from admin where userName=?";
        try
        
            prepar = conn.prepareStatement(sql);
            prepar.setString(1, userName);
            res = prepar.executeQuery();
            while (res.next())
            
                ub.setUserName(res.getString("userName"));
                ub.setUserPwd(res.getString("userPwd"));
                ub.setUserId(i);

            
            i++;
        

        catch (SQLException ex)
        
            ex.printStackTrace();
        

        return ub;
    

    public boolean checkParameter(String para) //¹ýÂË·Ç·¨×Ö·û
    
        int flag = 0;
        flag += para.indexOf("'") + 1;
        flag += para.indexOf(";") + 1;
        flag += para.indexOf("1=1") + 1;
        flag += para.indexOf("|") + 1;
        flag += para.indexOf("<") + 1;
        flag += para.indexOf(">") + 1;
        if (flag != 0)
        
            System.out.println("Ìá½»ÁË·Ç·¨×Ö·û!!!");
            return false;
        
        return true;
    

    public ArrayList selectCDBean(String selectValue, int page, int count) //²éѯ·ÖÒ³
    
        ArrayList list = new ArrayList();
        if (!checkParameter(selectValue))
        
            selectValue = "";
        
        try
        
            proc = conn.prepareCall("call proc_page(?,?,?,?)");
            proc.setInt(1, page);
            proc.setInt(2, count);
            proc.setString(3, selectValue);
            proc.registerOutParameter(4, Types.INTEGER); //OUTPUT²ÎÊý ·µ»Ø½á¹¹¹²¶àÉÙÒ³
            res = proc.executeQuery(); //½ÓÊÕ´æ´¢¹ý³ÌµÄ½á¹û¼¯
            while (res.next()) //ÌáÈ¡½á¹û¼¯µÄÿÌõ¼Ç¼
            
                CDBean cb = new CDBean();
                cb.setCdAlbum(res.getString("CDalbum"));
                cb.setCdCompany(res.getString("CDcompany"));
                cb.setCdName(res.getString("CDname"));
                cb.setCdId(res.getLong("CDid"));
                cb.setCdType(getCDType(res.getInt("CDtypeId")));
                list.add(cb);
            
            PAGECOUNT = proc.getInt(4);

        
        catch (SQLException ex)
        
            ex.printStackTrace();
        

        return list;

    

    public String getCDType(int cdtypeId)
    
        ResultSet res1=null;
        try
        

            java.sql.PreparedStatement prepar1 = conn.prepareStatement(
                "select display from CDtype where CDtypeId=?");
            prepar1.setLong(1, cdtypeId);
            res1 = prepar1.executeQuery();
            res1.next();
            return res1.getString("display");
        
        catch (SQLException ex)
        

            return null;
        
    
    public boolean setCDBean(CDBean cb)
    
        if (!checkParameter(cb.getCdName() + cb.getCdCompany() + cb.getCdAlbum() +
                            cb.getCdType()))
        
            return false;
        

        boolean flag = false;
        String sql =
            "update CDinfo set CDname=?,CDcompany=?,CDalbum=?,CDtypeId=? where CDid=?";
        try
        
            prepar = conn.prepareStatement(sql);
            prepar.setString(1, cb.getCdName());
            prepar.setString(2, cb.getCdCompany());
            prepar.setString(3, cb.getCdAlbum());
            prepar.setInt(4, Integer.parseInt(cb.getCdType()));
           // prepar.setLong(5, cb.getCdId());
            int result = prepar.executeUpdate();
            if (result > 0)
            
                flag = true;
            
            else
            
                flag = false;
            

        
        catch (Exception ex)
        
            flag = false;
            ex.printStackTrace();
        
        return flag;
    

    public CDBean getCDBean(long id)
    

        CDBean cb = new CDBean();
        int i=1;
        String sql = "select * from CDinfo where CDid=?";
        try
        
            prepar = conn.prepareStatement(sql);
            prepar.setLong(1, id);
            res = prepar.executeQuery();
            while (res.next())
            
                cb.setCdAlbum(res.getString("CDalbum"));
                cb.setCdCompany(res.getString("CDcompany"));
                cb.setCdName(res.getString("CDname"));
                cb.setCdId(i);
                cb.setCdType(getCDType(res.getInt("CDtypeId")));

            
              i++;
        
        catch (SQLException ex)
        
            ex.printStackTrace();
        
        return cb;
    

    public boolean deleteCDBean(long id)
    
        boolean flag = false;
        String sql = "delete from CDinfo where CDid=?";
        try
        
            prepar = conn.prepareStatement(sql);
            prepar.setLong(1, id);
            int result = prepar.executeUpdate();
            if (result > 0)
            
                flag = true;
            
            else
            
                flag = false;
            
        
        catch (Exception ex)
        
            flag = false;
            ex.printStackTrace();
        
        return flag;
    

    public boolean addCDBean(CDBean cb)
    
        boolean flag = false;
        if (!checkParameter(cb.getCdName() + cb.getCdCompany() + cb.getCdAlbum() + cb.getCdId()+
                            cb.getCdType()))
        
            return false;
        
        String sql = "insert into CDinfo values(?,?,?,default,?)";
        try
        
            this.prepar = conn.prepareStatement(sql);
            prepar.setString(1, cb.getCdName());
            prepar.setString(2, cb.getCdCompany());
            prepar.setString(3, cb.getCdAlbum());
            prepar.setInt(4, Integer.parseInt(cb.getCdType()));
            int result = prepar.executeUpdate();
            if (result > 0)
            
                flag = true;
            
            else
            
                flag = false;
            

        
        catch (Exception ex)
        
            flag = false;
            ex.printStackTrace();
        
        return flag;
    

    public boolean setUserBean(UserBean ub)
    
        boolean flag = false;
        String sql = "update admin set userPwd=? where userId=?";
        try
        
            if (!checkParameter(ub.getUserPwd()))
            
                return false;
            
            this.prepar = conn.prepareStatement(sql);
            prepar.setString(1, ub.getUserPwd());
            prepar.setLong(2, ub.getUserId());
            int result = prepar.executeUpdate();
            if (result > 0)
            
                flag = true;
            
            else
            
                flag = false;
            
        
        catch (Exception ex)
        
            flag = false;
            ex.printStackTrace();
        
        return flag;
    

    public boolean addUserBean(UserBean ub)
    
        boolean flag = false;
        String sql = "insert into admin(userName,userPwd) values(?,?)";
        //int i=1;
        if (!checkParameter(ub.getUserPwd() + ub.getUserName()+ub.getUserId()))
        
            return false;
        
        if (hasUser(ub.getUserName()))
        
            return false;
        
        try
        
            prepar = conn.prepareStatement(sql,prepar.RETURN_GENERATED_KEYS);
            prepar.setString(1, ub.getUserName());
            prepar.setString(2, ub.getUserPwd());
           // prepar.setLong(3,ub.getUserId());
            int result = prepar.executeUpdate();
            if (result > 0)
            
                flag = true;
            
            else
            
                flag = false;
            
          //  i++;

        
        catch (Exception ex)
        
            flag = false;
            ex.printStackTrace();
        

        return flag;
    

    public boolean hasUser(String userName)
    
        boolean flag = true;
        String sql = "select count(*) from admin where userName=?";
        try
        
            prepar = conn.prepareStatement(sql);
            prepar.setString(1, userName);
            res = prepar.executeQuery();
            res.next();
            int result = res.getInt(1);
            if (result > 0)
            
                flag = true;
            
            else
            
                flag = false;
            
        
        catch (SQLException ex)
        
            ex.printStackTrace();
            flag = true;
        

        return flag;
    



【问题讨论】:

【参考方案1】:
public class DataBean
    private Connection conn = null;
    private ResultSet res = null;
    // ...
    private Statement stmt=null;
    private java.sql.PreparedStatement prepar = null;
    private java.sql.CallableStatement proc = null;
    // ...

这是错误的。您不应该在类级别声明它们。您应该在方法级别声明它们。以你的getUser() 方法为例,它必须是这样的:

public UserBean getUser(String userName) 
    Connection connection = null;
    PreparedStatement statement = null;
    ResultSet resultSet = null;
    // ...

然后,您需要确保以相反的顺序关闭 finally 块中的资源,因为您在同一个 try 块中获取它们。这是根据推荐的 JDBC 习惯用法对 getUser() 方法的完整重写:

public UserBean getUser(String userName) throws SQLException 
    Connection connection = null;
    PreparedStatement statement = null;
    ResultSet resultSet = null;
    UserBean userBean = null;

    String sql = "select userId, userName, userPwd from admin where userName = ?";

    try 
        connection = dataSource.getConnection();
        statement = connection.prepareStatement(sql);
        statement.setString(1, userName);
        resultSet = statement.executeQuery();

        if (resultSet.next()) 
            userBean = new UserBean();
            userBean.setUserId(resultSet.getInt("userId");
            userBean.setUserName(resultSet.getString("userName"));
            userBean.setUserPwd(resultSet.getString("userPwd"));
        
     finally 
        if (resultSet != null) try  resultSet.close();  catch (SQLException logOrIgnore) 
        if (statement != null) try  statement.close();  catch (SQLException logOrIgnore) 
        if (connection != null) try  connection.close();  catch (SQLException logOrIgnore) 
    

    return userBean;

(请注意,我将用户 ID 固定为数据库列字段,使其成为自动增量 PK,还请注意,我将 while 固定为 if - 只有 1 个用户这个名字,对吧?-,还要注意它返回null,当没有已知用户时可以方便地检查)

另见:

How often should Connection, Statement and ResultSet be closed in JDBC? JDBC MySql connection pooling practices to avoid exhausted connection pool Am I Using JDBC Connection Pooling?

您的 checkUsersLogin()checkParameter() 方法也很可疑,但这是另一个问题的主题。

【讨论】:

不客气。由于您是新来的,请不要忘记标记已接受的答案,这有助于(最)解决您的问题。另见meta.stackexchange.com/questions/5234/…【参考方案2】:

您必须使用相反的顺序关闭每个可关闭对象。 如果开单: 连接 PreparedStatement 结果集

那么关闭订单将是: 结果集 PreparedStatement 连接

您可以将其放入 finally 块中。这是一般规则。

【讨论】:

感谢@e-zinc,所以我应该在上面的每个方法中添加 finally 这些对象在所有情况下都必须关闭,包括异常,使用 finally 块很容易做到,当然你必须在之前检查它们是否为空。【参考方案3】:

您应该使用 Java 7 try-with-resources。是的,当你问这个问题时它已经出来了! ;)

类似的东西:

try (Connection con = // get connection... ;
     PreparedStatement ps = // create prepared statement... ; 
     ResultSet rs = ps.executeQuery()) 
     // Process the result set...
     // The AutoCloseable resources 'conn', 'ps', and 'rs' will be cleaned up at the end:
     // .close() will be called on all of them whether an exception occurs or not
 catch (SQLException e) 
    // TODO(mykey): Do not swallow me!!

注意:请参阅AutoCloseable .close() 的文档

【讨论】:

以上是关于如何在以下几种方法中关闭 resultSet、prepareStatement、conn 以避免 rs 关闭和连接池被卡住的主要内容,如果未能解决你的问题,请参考以下文章

04在 PR 中关闭 issue

为什么我们应该在JDBC中关闭连接?如果我们不这样做,会发生什么

java中关闭窗口的方法

如何在 macOS 中关闭 OpenCV 窗口(Python 3)?

winform中关闭退出和打开新窗口的几种方式

如何在 Android SearchView 中关闭键盘?