使用网上流传的一个数据库连接池在Proxy.newProxyInstance处引起 java.lang.ClassCastException 问题的解决方法

getFreeConnection 方法的代码如下:

    public Connection getFreeConnection() {
        // 返回数据库连接 conn 的接管类,以便截住 close 方法
        Connection conn2 = null;
        if (conn instanceof Connection) {
            conn2 = (Connection) Proxy.newProxyInstance(conn.getClass().getClassLoader(), conn.getClass().getInterfaces(), this);
        }
        return conn2;
    }
 
用 MySQLv5 版本的数据库驱动没有问题,使用 MySQLv6 和 Oracle 的数据库驱动时候报如下错误:
java.lang.ClassCastException: $Proxy0 cannot be cast to java.sql.Connection
        at dbpoolImpl._Connection.getFreeConnection(_Connection.java:126)
        at dbpoolImpl.ConnectionFactory.getFreeConnection(ConnectionFactory.java:113)
        at dbpoolImpl.DataBase_Task.checkDataBase(DataBase_Task.java:83)
        at dbpoolImpl.DataBase_Task.ProcessHandle(DataBase_Task.java:227)
 
在网上搜索很久,终于找到解决方法,将创建代理类的语句改一下“
conn2 = (Connection) Proxy.newProxyInstance(conn.getClass().getClassLoader(), new Class[]{Connection.class}, this);
 
原因就在于 conn.getClass().getInterfaces() 方法出来的是 Class 类数组,此数组的第一个元素必须是 Connection 才能把创建的代理类转为 Connection 对象,大家可以使用下面的语句
        Class[] interfaces = conn.getClass().getInterfaces();
        for (Class c : interfaces) {
            System.out.println(c.getCanonicalName());
        }
打印一下使用不同数据库驱动时 conn.getClass().getInterfaces() 里面都有啥元素