JAVA写入Access数据库并检索和索引

时间:2012-11-22 20:29:12

标签: java ms-access resultset

我正在将数据从Java写入Windows 32位的Access数据库。当我写一个记录时,我需要检索行ID /主键,以便我可以a)轻松更新记录,如果我想和b)交叉引用其他数据到该记录。

当在C中做类似的事情时,我可以创建一个可更新的游标,它允许我写一个新记录并同时检索行ID。使用Java,看起来我应该能够做到这一点,但它会抛出以下代码的异常。

    con = openAccessDatabase();
    String selectString = "SELECT ID, RunCount FROM SpeedTable";
    try {
        PreparedStatement selectStatement = con.prepareStatement(selectString, 
                ResultSet.TYPE_SCROLL_INSENSITIVE,
                ResultSet.CONCUR_UPDATABLE);
        ResultSet idResult = selectStatement.executeQuery();
        int id;
        for (int i = 0; i < nWrites; i++) {
            idResult.moveToInsertRow();
            idResult.updateObject(1, null); // this line makes no difference whatsoever !
            idResult.updateInt(2, i);
            idResult.insertRow(); // throws java.sql.SQLException: [Microsoft][ODBC Microsoft Access Driver]Error in row
            id = idResult.getInt(1);
        }
        selectStatement.close();
    } catch (SQLException e) {
        e.printStackTrace();
    }

我唯一能做的就是写一条新记录,然后运行一个不同的查询来获取Row id ...

    String insertString = "INSERT INTO SpeedTable (RunCount) VALUES (?)";
    String idString = "SELECT ID FROM SpeedTable ORDER BY ID DESC";
    //      
    try {
        ResultSet idResult = null;
        PreparedStatement preparedStatement, idStatement;
        preparedStatement = con.prepareStatement(insertString,
                ResultSet.TYPE_FORWARD_ONLY,
                ResultSet.CONCUR_READ_ONLY);
        idStatement = con.prepareStatement(idString, 
                ResultSet.TYPE_FORWARD_ONLY,
                ResultSet.CONCUR_READ_ONLY);

        for (int i = 0; i < nWrites; i++) {
            // write the data into the database
            preparedStatement.setInt(1, i);
            preparedStatement.execute();
            // re-run the query to get the index back from the database. 
            idResult = idStatement.executeQuery();
            idResult.next();
            int lastIndex = idResult.getInt(1);
            idResult.close();
        }

这种方法有效,但当表格中有超过10个1000的记录时,它变得非常慢。如果程序的两个部分同时开始写入(不太可能但不是不可能),则还存在返回错误ID的风险。

我知道至少有一个建议是不使用Java或不使用Access,但它们不是选项。它也是免费开源软件包的一部分,所以我不愿意支付任何费用。编写我自己的C JNI接口,它提供了我的应用程序所需的基本功能,甚至不那么吸引人了。

1 个答案:

答案 0 :(得分:0)

不确定这是否适用于MS Access,但您可以尝试:

st.executeUpdate("INSERT INTO SpeedTable (RunCount) VALUES (1000)", Statement.RETURN_GENERATED_KEYS);
ResultSet rs = st.getGeneratedKeys();
rs.next();
long id = rs.getLong(1);