MYSQL死锁问题java.sql.SQLException: Lock wait timeout exceeded; try restarting transaction

  • Post author:
  • Post category:java


数据库死锁

分类

MySQL有三种锁的级别:页级、表级、行级。

  • 表级锁:开销小,加锁快;不会出现死锁;锁定粒度大,发生锁冲突的概率最高,并发度最低。

  • 行级锁:开销大,加锁慢;会出现死锁;锁定粒度最小,发生锁冲突的概率最低,并发度也最高。

  • 页面锁:开销和加锁时间界于表锁和行锁之间;会出现死锁;锁定粒度界于表锁和行锁之间,并发度一般算法:

而产生死锁的原因/场景如下:


原理:

就像多线程死锁一样,比如线程ab都想写文件t,a操作完T后没有释放锁,b就没法读T了,而且a自己也没法继续下去。


场景

:

1、在同一事务内先后对同一条数据进行插入和更新操作

2、多台服务器操作同一数据库

3、瞬时出现高并发现象,spring事务造成数据库死锁,后续操作超时抛出异常

4、sql语句执行后未commit

今天我遇到的是第四个场景,

sql语句执行后未commit造成死锁。

java在使用jdbc时,可以设置conn.setAutoCommit(boolean b)函数关闭自动提交(b=false),然后我在关闭自动提交后,没有手动提交,表当然等着你提交了,可是你还没提交,后面的操作又来了,怎么办呢?就报错了:锁等待超时,请重启事务。奈何我一直没有提交,又如何开始事务?

java.sql.SQLException: Lock wait timeout exceeded; try restarting transaction
    synchronized public static void updateTableUserProfile(UserProfile userProfile){
        // 获取连接
        Connection conn = DBCtrl.getConnection();
        // 更新语句
        String sql = "UPDATE user_profile" +
                " set user_name=?, age=?, locate=?, self_intro=?" +
                " where id=?";
        try {
            PreparedStatement ptmt = conn.prepareStatement(sql);
            ptmt.setString(1, userProfile.GetUsername());
            ptmt.setInt(2, userProfile.GetAge());
            ptmt.setString(3, userProfile.GetLocate());
            ptmt.setString(4, userProfile.GetSelfIntro());
            ptmt.setInt(5, userProfile.GetId());
            System.out.println(userProfile.GetAge() + " " +userProfile.GetSelfIntro()+ " "+userProfile.GetId());
            ptmt.execute();
            conn.commit();// 我没有写这句 =========================================
            System.out.println("用户[" + userProfile.GetId() +"]成功从数据库更新基本信息.........Yes");
        }catch(SQLException e){
            e.printStackTrace();
            System.out.println("用户[" + userProfile.GetId() +"]成功从数据库更新基本信息.........No");
        }
    }



版权声明:本文为aDiligentCigarette原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。