Java操作数据库(完,行级锁,for update)

14 篇文章 1 订阅
5 篇文章 1 订阅

不知不觉,JDBC就要和大家说byebye了,下面就让我们来看看JDBC末尾的内容锁的概率吧!

目录

 悲观锁(也叫行级锁)

使用悲观锁(在sql语句中使用)

完整代码

测试代码

 结论


 悲观锁(也叫行级锁)

在本次事务的执行过程当中,你指定的记录被查询,在我查询的过程当中记录就会被锁定,任何人,任何事务都不能对我指定查询数据进行修改操作(不能改,但是可以看),直到我都查询结束。

使用悲观锁(在事务中sql语句中使用)

//sql指令
            String sql = "select * from  t_shuihuo where id < ? for update ";

完整代码

package com.luosf.jdbc;

import com.luosf.jdbc.utils.JdbcUtil;

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;

/**
 * JDBC中锁的使用
 * for updata
 */
public class JdbcLock {
    public static void main(String[] args) {
        Connection conn = null;
        PreparedStatement stat = null;
        ResultSet res = null;
        try {
            //创建驱动

            //获取数据库对象
             conn = JdbcUtil.getConnection();

            //sql指令
            String sql = "select * from  t_shuihuo where id < ? for update ";

            conn.setAutoCommit(false);//开启事务

            //3,sql语句进行编译
            stat = conn.prepareStatement(sql);

            //给占位符填充值
            //JDBC下标从1开始的
            stat.setInt(1,16); //1,代表第一个问号

            Thread.sleep(1000*10); //模拟访问时间

            //4,执行sql
            res = stat.executeQuery();

            //5,处理查询结果集
            while (res.next()){
                int id = res.getInt("id");
                String name = res.getString("name");
                String nickname = res.getString("nickname");
                System.out.println("id :"+ id + "  name :" +name + "  昵称 :"+nickname);
            }
            conn.commit();//提交事务
        } catch (SQLException throwables) {
            try {
                if (conn != null){
                    conn.rollback(); //回滚事务
                }

            } catch (SQLException e) {
                e.printStackTrace();
            }
            throwables.printStackTrace();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally{
            //释放资源
            JdbcUtil.close(conn,stat,res);
        }
    }
}

测试代码

package com.luosf.jdbc;

import com.luosf.jdbc.utils.JdbcUtil;

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.SQLException;

/**
 * 检测锁
 */
public class JdbcLockTest {
    public static void main(String[] args) {
        Connection conn = null;
        PreparedStatement stat = null;

        try {
            //获取驱动

            //获取数据库链接对象
            conn = JdbcUtil.getConnection();

            //开启事务
            conn.setAutoCommit(false);

            //锁开始后进行修改数据
            String sql = "update t_shuihuo set name = '小罗' where id  = ?  ";
            stat = conn.prepareStatement(sql);

            stat.setInt(1,10); //1,代表第一个问号

            int cunt = stat.executeUpdate();
            System.out.println("更新了"+cunt+"条数据");

            conn.commit();//提交事务
        } catch (SQLException throwables) {
            try {
                if (conn != null){
                    conn.rollback();
                }
            } catch (SQLException e) {
                e.printStackTrace();
            }
            throwables.printStackTrace();
        } finally {
            //释放资源
           JdbcUtil.close(conn,stat,null);
        }
    }
}

需要等锁等待时间完成才能进行修改

 结论

在MySQL当中:

在执行“select ... from ....whrer ...for update ”对,MySQL进行row lock(行锁) 还是 table lock(表锁),取决于是否使用索引(如主键,unique字段),则为row lock(行锁),否则为 table lock(表锁),没有查找到数据为无锁,当使用“<>” 或者“like”时,索引会失效,进行 table lock(表锁)。

简单点来说就是for update最好锁 主键或者unique字段,锁其他字段会导致整张表被锁。导致性能的降低

请谨慎使用!!!

  • 9
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 6
    评论
Java可以使用JDBC API来访问关系型数据库,并且可以使用JDBC API来管理数据库中的行级。 要在Java中使用数据库行级,请按照以下步骤进行操作: 1. 获取数据库连接。使用JDBC API中的DriverManager类和Connection接口,可以连接到数据库。 2. 开启事务。在Java中使用数据库行级时,通常需要使用事务。使用Connection接口的setAutoCommit方法将自动提交设置为false,然后使用Connection接口的beginTransaction方法开始一个事务。 3. 获取行级。在Java中获取行级,需要使用SELECT ... FOR UPDATE语句,例如: ``` String sql = "SELECT * FROM table_name WHERE id = ? FOR UPDATE"; PreparedStatement stmt = connection.prepareStatement(sql); stmt.setInt(1, id); ResultSet rs = stmt.executeQuery(); ``` 注意,FOR UPDATE语句将定SELECT语句返回的行,直到事务提交或回滚为止。 4. 更新数据。在获取行级之后,可以对数据库中的数据进行更新操作。使用PreparedStatement接口的executeUpdate方法来执行更新操作。 5. 提交事务。在更新操作成后,使用Connection接口的commit方法提交事务。如果发生任何错误,可以使用Connection接口的rollback方法回滚事务。 整的示例代码如下所示: ``` Connection connection = null; PreparedStatement stmt = null; ResultSet rs = null; try { connection = DriverManager.getConnection(url, username, password); connection.setAutoCommit(false); String sql = "SELECT * FROM table_name WHERE id = ? FOR UPDATE"; stmt = connection.prepareStatement(sql); stmt.setInt(1, id); rs = stmt.executeQuery(); // 更新数据 String updateSql = "UPDATE table_name SET column_name = ? WHERE id = ?"; PreparedStatement updateStmt = connection.prepareStatement(updateSql); updateStmt.setString(1, "new value"); updateStmt.setInt(2, id); updateStmt.executeUpdate(); connection.commit(); } catch (SQLException e) { if (connection != null) { try { connection.rollback(); } catch (SQLException ex) { ex.printStackTrace(); } } e.printStackTrace(); } finally { if (rs != null) { try { rs.close(); } catch (SQLException e) { e.printStackTrace(); } } if (stmt != null) { try { stmt.close(); } catch (SQLException e) { e.printStackTrace(); } } if (connection != null) { try { connection.close(); } catch (SQLException e) { e.printStackTrace(); } } } ``` 在上面的示例代码中,使用了一个SELECT ... FOR UPDATE语句来获取行级,并使用一个UPDATE语句来更新数据。在更新成后,使用commit方法提交事务。如果发生任何错误,将回滚事务。
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

韶光不负

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值