Java sql事务。我做错了什么?

Java sql事务。我做错了什么?,java,jdbc,transactions,Java,Jdbc,Transactions,我编写这个小测试的唯一目的是更好地理解jdbc中的事务。虽然我是根据文档做的,但是测试不希望正常工作 以下是表格结构: CREATE TABLE `default_values` ( `id` INT UNSIGNED NOT auto_increment, `is_default` BOOL DEFAULT false, PRIMARY KEY(`id`) ); 测试包含3个类: public class DefaultDeleter implements Runnable

我编写这个小测试的唯一目的是更好地理解jdbc中的事务。虽然我是根据文档做的,但是测试不希望正常工作

以下是表格结构:

CREATE TABLE `default_values` (
   `id` INT UNSIGNED NOT auto_increment,
   `is_default` BOOL DEFAULT false,
   PRIMARY KEY(`id`)
);
测试包含3个类:

public class DefaultDeleter implements Runnable
{

    public synchronized void deleteDefault() throws SQLException
    {
        Connection conn = null;
        Statement deleteStmt = null;
        Statement selectStmt = null;
        PreparedStatement updateStmt = null;
        ResultSet selectSet = null;

        try
        {
            conn = DriverManager.getConnection("jdbc:mysql://localhost/xtest", "root", "");
            conn.setAutoCommit(false);
            conn.setTransactionIsolation(Connection.TRANSACTION_SERIALIZABLE);

            // Deleting current default entry
            deleteStmt = conn.createStatement();
            deleteStmt.executeUpdate("DELETE FROM `default_values` WHERE `is_default` = true");

            // Selecting first non default entry
            selectStmt = conn.createStatement();
            selectSet = selectStmt.executeQuery("SELECT `id` FROM `default_values` ORDER BY `id` LIMIT 1");

            if (selectSet.next())
            {
                int id = selectSet.getInt("id");

                // Updating found entry to set it default
                updateStmt = conn.prepareStatement("UPDATE `default_values` SET `is_default` = true WHERE `id` = ?");
                updateStmt.setInt(1, id);
                if (updateStmt.executeUpdate() == 0)
                {
                    System.err.println("Failed to set new default value.");
                    System.exit(-1);
                }
            }
            else
            {
                System.err.println("Ooops! I've deleted them all.");
                System.exit(-1);
            }

            conn.commit();
            conn.setAutoCommit(true);
        }
        catch (SQLException e)
        {
            try { conn.rollback(); } catch (SQLException ex)
            {
                ex.printStackTrace();
            }

            throw e;
        }
        finally
        {
            try { selectSet.close(); } catch (Exception e) {}
            try { deleteStmt.close(); } catch (Exception e) {}
            try { selectStmt.close(); } catch (Exception e) {}
            try { updateStmt.close(); } catch (Exception e) {}
            try { conn.close(); } catch (Exception e) {}
        }
    }

    public void run()
    {
        while (true)
        {
            try
            {
                deleteDefault();
            }
            catch (SQLException e)
            {
                e.printStackTrace();
                System.exit(-1);
            }

            try
            {
                Thread.sleep(20);
            }
            catch (InterruptedException e) {}
        }
    }

}

public class DefaultReader implements Runnable
{

    public synchronized void readDefault() throws SQLException
    {
        Connection conn = null;
        Statement stmt = null;
        ResultSet rset = null;

        try
        {
            conn = DriverManager.getConnection("jdbc:mysql://localhost/xtest", "root", "");

            conn.setAutoCommit(false);
            conn.setTransactionIsolation(Connection.TRANSACTION_SERIALIZABLE);

            stmt = conn.createStatement();
            rset = stmt.executeQuery("SELECT * FROM `default_values` WHERE `is_default` = true");

            int count = 0;
            while (rset.next()) { count++; }

            if (count == 0)
            {
                System.err.println("Default entry not found. Fail.");
                System.exit(-1);
            }
            else if (count > 1)
            {
                System.err.println("Count is " + count + "! Wtf?!");
            }

            conn.commit();
            conn.setAutoCommit(true);
        }
        catch (SQLException e)
        {
            try { conn.rollback(); } catch (Exception ex)
            {
                ex.printStackTrace();
            }

            throw e;
        }
        finally
        {
            try { rset.close(); } catch (Exception e) {}
            try { stmt.close(); } catch (Exception e) {}
            try { conn.close(); } catch (Exception e) {}
        }
    }

    public void run()
    {
        while (true)
        {
            try
            {
                readDefault();
            }
            catch (SQLException e)
            {
                e.printStackTrace();
                System.exit(-1);
            }

            try
            {
                Thread.sleep(20);
            }
            catch (InterruptedException e) {}
        }
    }

}

public class Main
{

    public static void main(String[] args)
    {
        try
        {
            Driver driver = (Driver) Class.forName("com.mysql.jdbc.Driver")
                    .newInstance();
            DriverManager.registerDriver(driver);

            Connection conn = null;
            try
            {
                conn = DriverManager.getConnection("jdbc:mysql://localhost/xtest", "root", "");
                System.out.println("Is transaction isolation supported by driver? " +
                        (conn.getMetaData()
                        .supportsTransactionIsolationLevel(
                        Connection.TRANSACTION_SERIALIZABLE) ? "yes" : "no"));
            }
            finally
            {
                try { conn.close(); } catch (Exception e) {}
            }

            (new Thread(new DefaultReader())).start();
            (new Thread(new DefaultDeleter())).start();

            System.in.read();
            System.exit(0);
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
    }

}
我已经编写了脚本,它用100k条记录填充表,其中一条记录是每次运行的默认记录。但每次我运行此测试时,输出是:

驱动程序是否支持事务隔离?对

未找到默认条目。失败


此代码有什么问题?

如果允许容器管理事务,可以执行以下操作:

@Resource
private UserTransaction utx;
然后在代码中使用它:

utx.begin();

// atomic operation in here

utx.commit();
那么您就不必担心事务管理的复杂性了


编辑:@Gris:是的,你说得对。我以为你在开发一个web应用程序。正如pjp所说,在这种情况下,spring是一个不错的选择。或者-取决于应用程序的大小和复杂性-您可以管理自己的事务。

我建议您添加一些断点,并逐步执行每个数据库操作,以检查它们是否按照您的预期执行。您可以在数据库服务器上打开会话,并设置事务隔离级别,以便读取未提交的数据


还要检查在MySql中对布尔类型的数值1使用“true”是否有效。

答案很简单:创建两个线程。它们完全独立运行。由于您没有以任何方式同步它们,因此无法确定哪一个先访问数据库。如果读卡器是第一个,那么删除器还没有开始,也不会有is_default==true的项,因为删除器还没有到达那么远

接下来,您已经完全隔离了两个事务Connection.TRANSACTION_SERIALIZABLE。这意味着,即使删除程序有机会更新数据库,读者也只能在关闭连接并打开新连接后才能看到它

如果不是这样,删除程序比读取器慢,因此在读取器查找记录时,记录更新为is_default==true的可能性很小


[编辑]现在您说,当测试开始时,应该有一个is_default==true的项目。在启动两个线程之前,请添加一个测试以确保情况确实如此。否则,您可能会发现错误的错误。

请确保您正在创建InnoDB表,MyISAM默认不支持事务。您可以将db create更改为:

CREATE TABLE `default_values` (
   `id` INT UNSIGNED NOT auto_increment,
   `is_default` BOOL DEFAULT false,
   PRIMARY KEY(`id`)
) Engine=InnoDB;

另一个例子:

有几点值得一提:

在测试之前填充数据库的脚本真的有效吗?尝试进行选择计数*。。。从Java代码内部检查这个可能听起来很愚蠢,但我以前犯过这个错误

不要到处执行System.exit,因为这会使代码很难测试-尽管看起来您没有默认==true记录,但查看删除程序的操作可能会很有趣


但若我理解正确,那个么它只有在应用程序在j2ee容器中运行时才可用。但我的目标是独立的j2se应用程序,而不是Web应用程序。作为成熟的j2ee的替代方案,您可以使用Spring实现DataSource TransactionManager。2 pjp:您可能是对的。我将查看文档,但我不确定从其他角度来看是否可以接受使用这样的框架。所有操作本身都是正确的。使用“true”也可以,它不是括号中的文本,而是内置常量。我首先检查了MySQL控制台中的所有查询。我感谢上面的所有评论对您的帮助——但我建议您从确保使用符合事务的表开始——然后Aaron Digulla提出了一个问题:哪个线程首先访问数据库。一旦使用了事务表,就可以在writer线程中设置睡眠,一次写入100行,然后确保delete在writer线程的中间事务中启动。你应该看到交易在做他们在那一点上做的事情。看起来我真的犯了这样一个愚蠢的错误。将引擎更改为InnoDB后,应用程序似乎运行良好。谢谢。关于线程-这是一个要求。测试作为一个可靠的应用程序工作,但最终的应用程序可能由几个完全独立的过程组成。关于事务——据我所知,完全隔离的事务应该提供封闭操作的原子性,以便内部的所有操作都作为一个实体操作进行处理—在我的例子中,删除、选择、更新。我错了吗?我使用脚本在每次运行之前清除并填充表。我也手动检查了启动条件。我看到一些答案开始谈论声明性事务和使用Spring,我很喜欢Spring,但它们只会混淆情况。这个问题是可以解决的,你会
更好地理解如何使用原始JDBC,如果您开始将其包装到更高级别的框架(如JEE或Spring)中,它就会变得模糊。我建议长期不要使用原始JDBC,除非您处于一个非常受限的环境中。它容易出错,而且是一个PITA。为什么您认为DefaultReader会成功?仅仅因为线程在删除程序之前启动,并不意味着它将在删除发生之前实际运行?请记住,MyISAM表不支持事务,InnoDB表支持事务。可序列化隔离级别可能会失败,如果有2个并发可序列化事务正在进行,dbs通常会使事务失败。