JAVA连mysql

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JAVA连mysql相关的知识,希望对你有一定的参考价值。

我的mysql用户名和密码都是root
以下两个代码:
1-
public class ConnectionDemo01
// 定义MySQL的数据库驱动程序
public static final String DBDRIVER = "org.gjt.mm.mysql.Driver" ;
public static void main(String args[])
try
Class.forName(DBDRIVER) ; // 加载驱动程序
catch(ClassNotFoundException e)
e.printStackTrace() ;


;
运行没问题
2-
import java.sql.Connection ;
import java.sql.DriverManager ;
import java.sql.SQLException ;
public class ConnectionDemo02
// 定义MySQL的数据库驱动程序
public static final String DBDRIVER = "org.gjt.mm.mysql.Driver" ;
// 定义MySQL数据库的连接地址
public static final String DBURL = "jdbc:mysql://localhost:3306/mldn" ;
// MySQL数据库的连接用户名
public static final String DBUSER = "root" ;
// MySQL数据库的连接密码
public static final String DBPASS = "mysqladmin" ;
public static void main(String args[])
Connection conn = null ; // 数据库连接
try
Class.forName(DBDRIVER) ; // 加载驱动程序
catch(ClassNotFoundException e)
e.printStackTrace() ;

try
conn = DriverManager.getConnection(DBURL,DBUSER,DBPASS) ;
catch(SQLException e)
e.printStackTrace() ;

System.out.println(conn) ; // 如果此时可以打印表示连接正常
try
conn.close() ; // 数据库关闭
catch(SQLException e)
e.printStackTrace() ;


;
编译正确
运行错误
错误代码:
com.mysql.jdbc.CommunicationsException: Communications link failure due to under
lying exception:

** BEGIN NESTED EXCEPTION **

java.net.SocketException
MESSAGE: java.net.ConnectException: Connection refused: connect

STACKTRACE:

java.net.SocketException: java.net.ConnectException: Connection refused: connect

at com.mysql.jdbc.StandardSocketFactory.connect(StandardSocketFactory.ja
va:156)
at com.mysql.jdbc.MysqlIO.<init>(MysqlIO.java:276)
at com.mysql.jdbc.Connection.createNewIO(Connection.java:2641)
at com.mysql.jdbc.Connection.<init>(Connection.java:1531)
at com.mysql.jdbc.NonRegisteringDriver.connect(NonRegisteringDriver.java
:266)
at java.sql.DriverManager.getConnection(DriverManager.java:582)
at java.sql.DriverManager.getConnection(DriverManager.java:185)
at ConnectionDemo02.main(ConnectionDemo02.java:21)

** END NESTED EXCEPTION **

Last packet sent to the server was 21 ms ago.
at com.mysql.jdbc.Connection.createNewIO(Connection.java:2707)
at com.mysql.jdbc.Connection.<init>(Connection.java:1531)
at com.mysql.jdbc.NonRegisteringDriver.connect(NonRegisteringDriver.java
:266)
at java.sql.DriverManager.getConnection(DriverManager.java:582)
at java.sql.DriverManager.getConnection(DriverManager.java:185)
at ConnectionDemo02.main(ConnectionDemo02.java:21)
null
Exception in thread "main" java.lang.NullPointerException
at ConnectionDemo02.main(ConnectionDemo02.java:27)

查看了Mysql的文档,以及Connector/J的文档以及在线说明发现,出现这种异常的原因是:Mysql服务器默认的“wait_timeout”是8小时,也就是说一个connection空闲超过8个小时,Mysql将自动断开该 connection。这就是问题的所在,在C3P0 pools中的connections如果空闲超过8小时,Mysql将其断开,而C3P0并不知道该connection已经失效,如果这时有 Client请求connection,C3P0将该失效的Connection提供给Client,将会造成上面的异常。
上网搜索,在MySQL的论坛上找到一个办法,就是如果在执行sql语句的时候发生了上述异常,就将sql语句重新执行一次。
试验发现,这个办法对这个使用spring+hibernate的服务无效。
进一步搜索发现,MySQL官方不推荐使用autoReconnect=true,参见http://bugs.mysql.com/bug.php?id=5020
需要另外找别的办法来解决这个问题。

由于问题产生的根本原因在于服务到数据库的连接长时间没活动,既然重新连接的办法无效,就可以尝试另外一种办法,就是反空闲。
自己写一个线程来反空闲的话,比较麻烦。
最后在网上找到一个办法。为hibernate配置连接池,推荐用c3p0,然后配置c3p0的反空闲设置idle_test_period,只要小于MySQL的wait timeout即可。
在hibernate.cfg.xml中增加下面几项:
<!-- configuration pool via c3p0-->
<property name="hibernate.connection.provider_class">org.hibernate.connection.C3P0ConnectionProvider</property>
<property name="c3p0.min_size">5</property>
<property name="c3p0.max_size">30</property>
<property name="c3p0.time_out">1800</property> <!-- seconds --><!-- default: 0 -->
<property name="c3p0.max_statement">50</property> <!-- default: 0 -->
<property name="c3p0.acquire_increment">1</property> <!-- default: 1 -->
<property name="c3p0.idle_test_period">120</property> <!-- seconds --><!-- default: 0 -->
<property name="c3p0.validate">true</property>

修改完后测试,问题解决。

--------------------------------------------------------

DBCP连接池说明:
driverClassName
url
username
password
上面四个分别是驱动,连接字符串,用户名和密码

maxActive 连接池支持的最大连接数
maxIdle 连接池中最多可空闲maxIdle个连接
minIdle 连接池中最少空闲maxIdle个连接
initialSize 初始化连接数目
maxWait 连接池中连接用完时,新的请求等待时间,毫秒
timeBetweenEvictionRunsMillis timeBetweenEvictionRunsMillis和minEvictableIdleTimeMillis一起使用,每

timeBetweenEvictionRunsMillis毫秒秒检查一次连接池中空闲的连接,把空闲时间超过minEvictableIdleTimeMillis毫秒的连接断开,直到连接池中的连接数到minIdle为止

minEvictableIdleTimeMillis 连接池中连接可空闲的时间,毫秒

removeAbandoned true,false,是否清理removeAbandonedTimeout秒没有使用的活动连接,清理后并没有放回连接池
removeAbandonedTimeout 活动连接的最大空闲时间
logAbandoned true,false,连接池收回空闲的活动连接时是否打印消息

minEvictableIdleTimeMillis,removeAbandonedTimeout这两个参数针对的连接对象不一样,minEvictableIdleTimeMillis针对连接池中的连接对象,removeAbandonedTimeout针对未被close的活动连接.

c3p0连接池说明:
driverClass
jdbcUrl
user
password

minPoolSize
maxPoolSize
initialPoolSize

acquireIncrement 池中没有空闲连接时,一次请求获取的连接数
maxIdleTime 池中连接最大空闲时间
acquireRetryAttempts 获取连接失败后,重新尝试的次数
acquireRetryDelay 尝试连接间隔时间,毫秒
checkoutTimeout 等待连接时间,0为无限等待,毫秒
DebugUnreturnedConnectionStackTraces true,false,是否收回未返回的活动连接
unreturnedConnectionTimeout 活动连接的时间.

jdbcurl建议不要使用autoReconnect=true。

----------------------------------------------------------------------

session.close();没有调用connection.close()吗?
如果你的Connection来自于连接池,他只不过被归还给池了,确实没有物理关闭,这是正常的结果。

若调用connection.close(), 此连接对象是关闭,还是没有关闭,只返回给了连接池 ?
那要看连接池的实现了。一般都是返回给连接池,因为新建连接的开销太大了。

创建一个SessionFactry就对应一个Connection,面SessionFactory中的Session是共享Connection .所以关闭Session对Connection没有影响的.

数据库连结池不过就是一个特殊的对象池而已。 对象池的作用就是避免你直接new资源性的对象,降低开销。把连结返回给连结池就是释放对该对象池中该Connection对象的引用,这样,这个对象可以给再次被别人使用。 你调用conn.close(),仅仅是释放了引用而已,不会关闭物理的连接。

connection对像在链接池中复写了close方法,所以并没有真正意义上的关闭。明白了吧。当然不同的链接池有不同的实现方法,connection只是一个接口,不同的链接池实现类是不一样的,只是我们感觉不到罢了。
参考技术A 应该是驱动包的问题 参考技术B Driver、数据库链接、数据库名称、用户名和密码都检查,再看看,数据库起了么 参考技术C 你看看你的代码21行...我看也是org.gjt.mm.mysql.Driver
问题
参考技术D 你是连接错误!第一个DRIVEQ地方就错了,应该是com.mysql.jdbc.driver 第5个回答  2010-05-18 1:保证你的MySql服务启动。
2:保证你访问的地址正确。

我看了你代码,你的这个是MLDN的吧?他的那个驱动包估计过时了。
现在用的都是
com.mysql.jdbc.Driver

无法使用java连接到mysql数据库[重复]

【中文标题】无法使用java连接到mysql数据库[重复]【英文标题】:failed to connect to mysql database using java [duplicate] 【发布时间】:2019-08-20 03:14:41 【问题描述】:

我正在尝试连接到 MySQL 数据库并在我的 java 代码中使用 sql 语句获取一些数据。但我无法连接到我的数据库并出现以下 SQL 异常。

错误:java.sql.SQLException: Access denied for user 'root'@'localhost' (using password: YES)

我的java代码如下:

Mysql_connecter.java

package mysql_connecter;

import java.sql.*;
/**
 *
 * @author kass
 */
public class Mysql_connecter 

    /**
     * @param args the command line arguments
     */
    public static void main(String[] args) 
    
        try
        
            Class.forName("com.mysql.jdbc.Driver");
            System.out.println("connecting.....");
            try (Connection con = DriverManager.getConnection("jdbc:mysql://localhost:3306/mydb","user","password")) 
            
                System.out.println("connected to mysql DB");
                Statement stmt=con.createStatement();

                ResultSet rs=stmt.executeQuery("select * from my_table");

                while(rs.next())
                
                    System.out.println(rs.getInt(1)+"  "+rs.getString(2)+"  "+rs.getString(3));
                
            
            catch(SQLException e)
             
                System.out.println("ERROR: ["+e+"]");
            
        
        catch(ClassNotFoundException e)
         
            System.out.println(e);
        
     

我已将 mysql jdbc 驱动程序添加到我的库路径中。我可以在我的代码中遗漏一些东西吗?

【问题讨论】:

请尝试使用 PHPMyAdmin 或 Heidisql 等 MysqlClient 访问数据库。如果您无法使用客户端访问 - 您不能期望它可以与 java 一起使用。 【参考方案1】:

您似乎没有为您的用户设置密码。 转到mysql 数据库并检查user 表以确认用户的用户名和密码。

【讨论】:

【参考方案2】:

这可能是使用用户名+密码组合但未在数据库端启用本机身份验证的问题。

您是否可以使用此命令更改用户的密码:

ALTER USER 'jeffrey'@'localhost'
IDENTIFIED WITH mysql_native_password
BY 'password';

https://dev.mysql.com/doc/refman/8.0/en/alter-user.html

有关本地身份验证如何工作的更多信息,请参阅

MySQL 8.0 Reference for Native Pluggable Authentication

或者如果您使用的是 MySQL 5.7,请参阅 MySQL 5.7 Reference for Native Pluggable Authentication

【讨论】:

以上是关于JAVA连mysql的主要内容,如果未能解决你的问题,请参考以下文章

MySQL常用命令

远程连接报错

MySQLMySQL增删改查与常见陷阱(MySQL专栏启动)

MySQLMySQL复制与高可用水平扩展架构实战(MySQL专栏启动)

mysql使用手册

MySQLMySQL参数调优与实战详解(调优篇)(实战篇)(MySQL专栏启动)