Как получить метаданные последовательности из JDBC? - PullRequest
5 голосов
/ 12 апреля 2011

Я пытаюсь получить различные виды метаданных моей базы данных Oracle из кода Java (используя базовый JDBC).Например, если я хочу получить список таблиц с суффиксом _FOO, я могу сделать что-то вроде:

Connection connection = dataSource.getConnection();
DatabaseMetaData meta = connection.getMetaData();
ResultSet tables = meta.getTables(connection.getCatalog(), null, "%_FOO", new String[] { "TABLE" });
// Iterate on the ResultSet to get information on tables...

Теперь я хочу получить все последовательности из моегобаза данных (например, вся последовательность с именем S_xxx_FOO).

Как бы я это сделал, поскольку я ничего не вижу в DatabaseMetaData, связанной с последовательностями?

Должен ли я выполнить запрос как select * from user_sequences?

Ответы [ 4 ]

3 голосов
/ 02 ноября 2011

У меня был тот же вопрос. Это довольно просто. Просто введите "SEQUENCE" в параметрах типов getMetaData (). GetTables ().

В вашем конкретном случае это будет что-то вроде:

meta.getTables (connection.getCatalog (), null, "% _FOO", new String [] {"SEQUENCE"});

3 голосов
/ 12 апреля 2011

Вы не можете сделать это через JDBC API, потому что некоторые базы данных (все еще) не поддерживают последовательности.

Единственный способ получить их - запросить системный каталог вашей СУБД (я полагаю, это Oracle в вашем случае, как вы упомянули user_sequences)

0 голосов
/ 02 декабря 2016

Учитывая, что последние версии драйверов Oracle JDBC (например, 12.1.0.2) не возвращают информацию о последовательности при вызове DatabaseMetaData#getTables с types, установленным на ["SEQUENCE"] Лучше всего самостоятельно выполнить необходимый запрос, например:

  SELECT o.owner AS sequence_owner,
       o.object_name AS sequence_name
  FROM all_objects o
  WHERE o.owner LIKE 'someOwnerPattern' ESCAPE '/'
    AND o.object_name LIKE 'someNamePattern' ESCAPE '/'
    AND o.object_type = 'SEQUENCE'
  ORDER BY 1, 2

... где someOwnerPattern и someNamePattern - это шаблоны SQL, подобные тем, которые вы использовали бы с оператором LIKE (например, % соответствует любому).

Это в основном то же самое, что и запрос, запускаемый самим драйвером, за исключением того, что он запрашивает объекты типа SEQUENCE.

0 голосов
/ 21 марта 2014

Вы можете использовать api hibernate для получения названия последовательности.см .: http://docs.jboss.org/hibernate/orm/3.2/api/org/hibernate/dialect/Dialect.html

В приведенном ниже примере вы можете увидеть, как использовать диалект для получения имен последовательностей

public static void main(String[] args) {
        Connection jdbcConnection = null;
        try {
            jdbcConnection = DriverManager.getConnection("", "", "");
            printAllSequenceName(jdbcConnection);
        } catch (SQLException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } finally {
            if(jdbcConnection != null) {
                try {
                    jdbcConnection.close();
                } catch (SQLException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
            }
        }
}

public static void printAllSequenceName(Connection conn) throws JDBCConnectionException, SQLException {
        DialectResolver dialectResolver = new StandardDialectResolver();
        Dialect dialect =  dialectResolver.resolveDialect(conn.getMetaData());

        if ( dialect.supportsSequences() ) {
            String sql = dialect.getQuerySequencesString();
            if (sql!=null) {

                Statement statement = null;
                ResultSet rs = null;
                try {
                    statement = conn.createStatement();
                    rs = statement.executeQuery(sql);

                    while ( rs.next() ) {
                        System.out.println("Sequence Name : " +  rs.getString(1));
                    }
                }
                finally {
                    if (rs!=null) rs.close();
                    if (statement!=null) statement.close();
                }

            }
        }
    }

Если вы не хотите использовать hibernate, то вам нужно создать ящикпользовательская последовательная реализация.

Пример кода для пользовательской реализации

interface SequenceQueryGenerator {
    String getSelectSequenceNextValString(String sequenceName);
    String getCreateSequenceString(String sequenceName, int initialValue, int incrementSize); 
    String getDropSequenceStrings(String sequenceName); 
    String getQuerySequencesString(); 
}


class OracleSequenceQueryGenerator implements SequenceQueryGenerator {

    @Override
    public String getSelectSequenceNextValString(String sequenceName) {
        return "select " + getSelectSequenceNextValString( sequenceName ) + " from dual";
    }

    @Override
    public String getCreateSequenceString(String sequenceName,
            int initialValue, int incrementSize) {
        return "create sequence " + sequenceName +  " start with " + initialValue + " increment by " + incrementSize;
    }

    @Override
    public String getDropSequenceStrings(String sequenceName) {
        return "drop sequence " + sequenceName;
    }

    @Override
    public String getQuerySequencesString() {
        return "select sequence_name from user_sequences";
    }

}


class PostgresSequenceQueryGenerator implements SequenceQueryGenerator {

    @Override
    public String getSelectSequenceNextValString(String sequenceName) {
        return "select " + getSelectSequenceNextValString( sequenceName );
    }

    @Override
    public String getCreateSequenceString(String sequenceName,
            int initialValue, int incrementSize) {
        return "create sequence " + sequenceName + " start " + initialValue + " increment " + incrementSize;
    }

    @Override
    public String getDropSequenceStrings(String sequenceName) {
        return "drop sequence " + sequenceName;
    }

    @Override
    public String getQuerySequencesString() {
        return "select relname from pg_class where relkind='S'";
    }

}

public void printSequenceName (SequenceQueryGenerator queryGenerator, Connection conn) throws SQLException {
        String sql = queryGenerator.getQuerySequencesString();
        if (sql!=null) {

            Statement statement = null;
            ResultSet rs = null;
            try {
                statement = conn.createStatement();
                rs = statement.executeQuery(sql);

                while ( rs.next() ) {
                    System.out.println("Sequence Name : " +  rs.getString(1));
                }
            }
            finally {
                if (rs!=null) rs.close();
                if (statement!=null) statement.close();
            }

        }
    }

public static void main(String[] args) {
        Connection jdbcConnection = null;
        try {
            jdbcConnection = DriverManager.getConnection("", "", "");
            printAllSequenceName(new OracleSequenceQueryGenerator(), jdbcConnection);
        } catch (SQLException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }finally {
            if(jdbcConnection != null) {
                try {
                    jdbcConnection.close();
                } catch (SQLException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
            }
        }
}
...