死锁在同一线程中打开两个 JDBC 事务



我正在通过JDBC测试一些数据库驱动程序。其中一个测试包括测试数据库的事务功能:

  1. 我打开了两个到同一个数据库的连接(使用 autoCommit=false(。
  2. 在连接 A 上,我在表中插入一行而不执行提交。
  3. 在连接 B 上,我希望还看不到该行。
  4. 在连接 A 上,我执行提交。
  5. 在连接 B 上,我希望看到该行。

该测试适用于Oracle 12,但对于HSQLDB,Derby或SQL Server等其他数据库,NO:它在步骤3的中间阻塞。

我想原因可能是与事务隔离参数有关,但是我在创建两个连接时尝试了所有可能的值,结果始终相同。

为什么会阻止?为什么甲骨文不阻止?

这是我的代码:

public class JdbcTest
{
    @Test
    public void twoTransactionsTest()
        throws SQLException,
        IOException
    {
        String tableName="dummy01";
        // 1. I open two connections to the same db (with autoCommit=false).
        try (Connection connectionA=createConnection(); Connection connectionB=createConnection())
        {
            createTable(connectionA, tableName);
            // 2. On connection A, I insert one row in a table without performing a commit.
            execute(connectionA, "INSERT INTO " + tableName + " VALUES(50, 'grecia')");
            // 3. On connection B, I expect not to see that row yet.
            int records=queryAndCountRows(connectionB, "SELECT id FROM " + tableName + " WHERE id=50");
            assertEquals(0, records);
            // 4. On connection A, I perform a commit.
            connectionA.commit();
            // 5. On connection B, I expect to see that row.
            records=queryAndCountRows(connectionB, "SELECT * FROM " + tableName + " WHERE id=50");
            assertEquals(1, records);
            dropTable(connectionA, tableName);
        }
    }
    private Connection createConnection()
        throws SQLException,
        IOException
    {
        String url="jdbc:hsqldb:demo.hsqldb";
        String user="demo";
        String password="";
        Connection connection=DriverManager.getConnection(url, user, password);
        connection.setAutoCommit(false);
        return connection;
    }
    private int queryAndCountRows(Connection connection, String sql)
        throws SQLException
    {
        try (PreparedStatement pst=connection.prepareStatement(sql))
        {
            try (ResultSet rs=pst.executeQuery())
            {
                int records=0;
                while (rs.next())
                {
                    records++;
                }
                return records;
            }
        }
    }
    private void execute(Connection connection, String sql)
        throws SQLException
    {
        try (Statement statement=connection.createStatement())
        {
            statement.execute(sql);
        }
    }
    private void createTable(Connection connection, String tableName)
        throws SQLException
    {
        try
        {
            execute(connection, "DROP TABLE " + tableName);
        }
        catch (SQLException e)
        {
            // If the table already exists, let's ignore this error.
        }
        execute(connection, "CREATE TABLE " + tableName + "(id NUMBER(5) NOT NULL, name VARCHAR2(100))");
    }
    private void dropTable(Connection connection, String tableName)
        throws SQLException
    {
        execute(connection, "DROP TABLE " + tableName);
    }
}

我的依赖项:

<dependency>
  <groupId>org.hsqldb</groupId>
  <artifactId>hsqldb</artifactId>
  <version>2.3.2</version>
</dependency>
<dependency>
  <groupId>com.oracle</groupId>
  <artifactId>ojdbc6</artifactId>
  <version>11.2.0</version>
</dependency>
<dependency>
  <groupId>com.microsoft.sqlserver</groupId>
  <artifactId>mssql-jdbc</artifactId>
  <version>6.1.0.jre8</version>
</dependency>

提前谢谢。

在 Oracle 中,数据库读取器从不等待写入器,除非在不寻常的情况下,您基本上可以忽略这些情况。 出于所有实际目的,读者从不等待作家。

在其他数据库中,情况并非总是如此。 例如,在HSQLDB中,默认锁定模式为2PL(两阶段锁定(。 在该模型中,写入表将获取该表上的独占锁,从而阻止其他会话读取该表。 它具有更复杂的锁定模型(例如,MVCC(,可以使用SET DATABASE TRANSACTION CONTROL命令进行设置。

这是一个很好的例子,说明为什么"数据库独立性"真的很难(对于我的钱来说,我从未设定过目标(。

隔离在这些

数据库系统中的实现方式不同,请点击此链接了解更多信息:

http://www.dba-in-exile.com/2012/11/isolation-levels-in-oracle-vs-sql-server.html

简而言之,Oracle以编写器不阻止读取器的方式实现它,但对于其他RDBMS来说并非如此。

如果将 HSQLDB 作为服务器运行并使用 MVCC,您将获得预期的结果。

在进程内使用 HSQLDB 时,必须为每个连接和 MVCC 使用单独的线程才能使其正常工作。

最新更新