Как выбрать данные из таблицы базы данных Derby в памяти?

Я создал таблицу базы данных в памяти, используя JavaDB/Derby. Я могу INSERT данные в таблицу. Но это не работает, когда я пытаюсь SELECT данные из таблицы.

Я создал таблицу с помощью:

DECLARE GLOBAL TEMPORARY TABLE SESSION.memtable (id int, name varchar(10))
NOT LOGGED

вставить данные с помощью:

INSERT INTO SESSION.memtable (id, name) VALUES (?,?)

и он возвращает 1 для rows affected. Я выбираю данные с помощью:

SELECT name FROM SESSION.memtable WHERE id = ?

но он возвращает пустой ResultSet (rs.next() ложно).

Что я делаю неправильно при использовании SELECT?

Я использую derby.jar, включенный в JDK7.

Вот мой код:

public class DBTest {
    public static void main(String[] args) {
        final int userId = 4;
        final String sql = "DECLARE GLOBAL TEMPORARY TABLE SESSION.memtable "+
                            "(id int, name varchar(10)) NOT LOGGED";
        final String inSQL = "INSERT INTO SESSION.memtable (id, name) "+
                             "VALUES (?,?)";
        final String selSQL = "SELECT name FROM SESSION.memtable WHERE id = ?"
        final String connURL = "jdbc:derby:memory:memdatabase;create=true";
        try(Connection conn = DriverManager.getConnection(connURL);) {
            try (PreparedStatement ps = conn.prepareStatement(sql);) {
                ps.execute();
            }
            try (PreparedStatement ps = conn.prepareStatement(inSQL);) {
                ps.setInt(1, userId);
                ps.setString(2, "Jonas");
                int rows = ps.executeUpdate();
                System.out.println(rows + " rows inserted.");
            }
            try (PreparedStatement ps = conn.prepareStatement(selSQL);) {
                ps.setInt(1, userId);
                try (ResultSet rs = ps.executeQuery();) {
                    String name;
                    if(rs.next()) {
                        name = rs.getString("name");
                    } else {
                        name = null;
                    }
                    System.out.println("Name: " + name);
                }
            }
        } catch (SQLException e) {e.printStackTrace();}}}

ОБНОВЛЕНИЕ

Если я изменю SELECT-запрос на:

SELECT t1.name AS name FROM SESSION.memtable t1 WHERE id = ?

Я получаю это сообщение об ошибке:

java.sql.SQLException: Operation 'DROP TABLE' cannot be performed on object 'MEMTABLE(ID, NAME)' because there is an open ResultSet dependent on that object.
    at org.apache.derby.impl.jdbc.SQLExceptionFactory40.getSQLException(Unknown Source)
    at org.apache.derby.impl.jdbc.Util.generateCsSQLException(Unknown Source)
    at org.apache.derby.impl.jdbc.TransactionResourceImpl.wrapInSQLException(Unknown Source)
    at org.apache.derby.impl.jdbc.TransactionResourceImpl.handleException(Unknown Source)
    at org.apache.derby.impl.jdbc.EmbedConnection.handleException(Unknown Source)
    at org.apache.derby.impl.jdbc.EmbedConnection.commitIfAutoCommit(Unknown Source)
    at org.apache.derby.impl.jdbc.ConnectionChild.commitIfAutoCommit(Unknown Source)
    at org.apache.derby.impl.jdbc.EmbedStatement.resultSetClosing(Unknown Source)
    at org.apache.derby.impl.jdbc.EmbedResultSet.movePosition(Unknown Source)
    at org.apache.derby.impl.jdbc.EmbedResultSet.next(Unknown Source)
    at DBTest.main(DBTest.java:29)
Caused by: java.sql.SQLException: Operation 'DROP TABLE' cannot be performed on object 'MEMTABLE(ID, NAME)' because there is an open ResultSet dependent on that object.
    at org.apache.derby.impl.jdbc.SQLExceptionFactory.getSQLException(Unknown Source)
    at org.apache.derby.impl.jdbc.SQLExceptionFactory40.wrapArgsForTransportAcrossDRDA(Unknown Source)
    ... 11 more
Caused by: ERROR X0X95: Operation 'DROP TABLE' cannot be performed on object 'MEMTABLE(ID, NAME)' because there is an open ResultSet dependent on that object.
    at org.apache.derby.iapi.error.StandardException.newException(Unknown Source)
    at org.apache.derby.impl.sql.conn.GenericLanguageConnectionContext.verifyNoOpenResultSets(Unknown Source)
    at org.apache.derby.impl.sql.GenericPreparedStatement.prepareToInvalidate(Unknown Source)
    at org.apache.derby.impl.sql.depend.BasicDependencyManager.coreInvalidateFor(Unknown Source)
    at org.apache.derby.impl.sql.depend.BasicDependencyManager.invalidateFor(Unknown Source)
    at org.apache.derby.impl.sql.conn.GenericLanguageConnectionContext.tempTablesAndCommit(Unknown Source)
    at org.apache.derby.impl.sql.conn.GenericLanguageConnectionContext.doCommit(Unknown Source)
    at org.apache.derby.impl.sql.conn.GenericLanguageConnectionContext.userCommit(Unknown Source)
    at org.apache.derby.impl.jdbc.TransactionResourceImpl.commit(Unknown Source)
    ... 6 more

Но Name: не печатается, так что все равно не работает.


person Jonas    schedule 11.11.2011    source источник


Ответы (2)


Здесь вы используете два понятия: одно — это Derby с опцией хранения In-Memors, а другое — временная таблица. Если вам нужна база данных In-Memors, вы все равно можете использовать обычные определенные таблицы ("CREATE TABLE..."). Это упростит задачу, так как вам не придется иметь дело с семантикой сеанса этих временных таблиц.

person eckes    schedule 10.08.2014

Вы действительно должны изучить Hibernate. Он не только отображает ваши классы, но и управляет неприятными вещами JDBC для вас, чтобы вы могли сосредоточиться на том, что вы хотите сделать. Он далек от совершенства, но помимо этого обеспечивает переносимость базы данных.

Я использую спящий режим через Spring, поэтому мой код будет выглядеть примерно так:

TempStorage temp = new TempStorage();
temp.setName("Bob");
getHibernateTemplate.insert(temp);
temp = getHibernateTemplate.find("SELECT name FROM TempStorage");
person Thom    schedule 11.11.2011
comment
Спящий режим - это не то, что я ищу, и я не понимаю, как это решает мою проблему. - person Jonas; 11.11.2011
comment
Использование гибернации поможет вам не увязнуть в особенностях языка базы данных. Хотя и добавляет пару своих. - person Thom; 11.11.2011
comment
Пожалуйста, объясните, как Hibernate решит мою проблему, и опубликуйте код. Как написать запрос на выборку из базы данных дерби в памяти, которая работает в Hibernate? - person Jonas; 11.11.2011
comment
Я подозреваю, что это даже невозможно сделать с помощью Hibernate + Derby. - person Jonas; 11.11.2011