Невозможно получить сгенерированные ключи через JDBC в хранилище данных SQL - PullRequest
0 голосов
/ 28 мая 2018

В моей базе данных хранилища данных SQL Azure есть такая таблица:

CREATE TABLE t_identity (
  id INTEGER IDENTITY (1, 1) NOT NULL,
  val INTEGER
)

Теперь, используя JDBC, я хочу вставить строку и извлечь сгенерированное значение идентификатора.Это будет работать в SQL Server и большинстве других баз данных:

try (Connection c = DriverManager.getConnection(url, properties);
    Statement s = c.createStatement()) {

    s.executeUpdate("insert into t_identity (val) values (1)",
        Statement.RETURN_GENERATED_KEYS);
    try (ResultSet rs = s.getGeneratedKeys()) {
        while (rs.next())
            System.out.println(rs.getObject(1));
    }
}

Но в хранилище данных SQL это не работает:

Exception in thread "main" com.microsoft.sqlserver.jdbc.SQLServerException: 'SCOPE_IDENTITY' is not a recognized built-in function name.
    at com.microsoft.sqlserver.jdbc.SQLServerException.makeFromDatabaseError(SQLServerException.java:264)
    at com.microsoft.sqlserver.jdbc.SQLServerStatement.getNextResult(SQLServerStatement.java:1585)
    at com.microsoft.sqlserver.jdbc.SQLServerStatement.doExecuteStatement(SQLServerStatement.java:876)
    at com.microsoft.sqlserver.jdbc.SQLServerStatement$StmtExecCmd.doExecute(SQLServerStatement.java:776)
    at com.microsoft.sqlserver.jdbc.TDSCommand.execute(IOBuffer.java:7385)
    at com.microsoft.sqlserver.jdbc.SQLServerConnection.executeCommand(SQLServerConnection.java:2750)
    at com.microsoft.sqlserver.jdbc.SQLServerStatement.executeCommand(SQLServerStatement.java:235)
    at com.microsoft.sqlserver.jdbc.SQLServerStatement.executeStatement(SQLServerStatement.java:210)
    at com.microsoft.sqlserver.jdbc.SQLServerStatement.executeUpdate(SQLServerStatement.java:2060)
    at SQLServer.main(SQLServer.java:65)

Другие методы (например, executeUpdate(String, String[])) тоже не работают, так как делегируют вышеуказанному.

Я понимаю, что хранилище данных SQL не поддерживает SCOPE_IDENTITY() и подобные функции ,который, кажется, используется за кулисами драйвером mssql-jdbc:

<dependency>
    <groupId>com.microsoft.sqlserver</groupId>
    <artifactId>mssql-jdbc</artifactId>
    <version>6.5.2.jre8-preview</version>
</dependency>

Но есть ли обходной путь?

Примечания:

Ответы [ 2 ]

0 голосов
/ 29 мая 2018

Можете ли вы описать ваш бизнес-пример более подробно?Если вы используете JOOQ, я думаю, что у вас может быть несовпадение технологий.

Хранилище данных SQL Azure не предназначено для транзакционных рабочих нагрузок.Если JOOQ просто поддерживает несколько ссылочных строк, то я не вижу проблемы, но если это ваша основная стратегия загрузки данных, вы не получите хорошего результата.

Вот некоторые чтения, которые могут помочь:

Примеры использования и анти-шаблоны: https://blogs.msdn.microsoft.com/sqlcat/2017/09/05/azure-sql-data-warehouse-workload-patterns-and-anti-patterns/

Модели решений (выделите диаграммы): https://azure.microsoft.com/en-us/services/sql-data-warehouse/

Рекомендации по загрузке данных: https://docs.microsoft.com/en-us/azure/sql-data-warehouse/guidance-for-loading-data

Шаблоны и стратегии загрузки данных: https://blogs.msdn.microsoft.com/sqlcat/2017/05/17/azure-sql-data-warehouse-loading-patterns-and-strategies/

0 голосов
/ 28 мая 2018

Обходной путь должен добавить к вашему запросу вставки, затем выберите идентификатор.

insert into t_identity
(val)
select 'my value'
where not exists
(
select 1
from t_identity
where val = 'my value'
)

select id
from t_dan_identity
where val = 'my value'
...