Выполнить оператор SQL через JDBC с привязкой CLOB
У меня есть следующий запрос (журнал столбца типа CLOB):
UPDATE table SET log=? where id=?
Приведенный выше запрос работает нормально при использовании метода setAsciiStream для помещения значения длиннее 4000 символов в столбец журнала.
Но вместо замены значения я хочу добавить его, поэтому мой запрос выглядит так:
UPDATE table SET log=log||?||chr(10) where id=?
Вышеуказанный запрос больше не работает, и я получаю следующую ошибку:
java.sql.SQLException: ORA-01461: can bind a LONG value only for insert into a LONG column
2 ответа
Мне кажется, что вы должны использовать блок PL/SQL, чтобы делать то, что вы хотите. Следующее работает для меня, предполагая, что есть запись с id
1:
import oracle.jdbc.OracleDriver;
import java.sql.*;
import java.io.ByteArrayInputStream;
public class JDBCTest {
// How much test data to generate.
public static final int SIZE = 8192;
public static void main(String[] args) throws Exception {
// Generate some test data.
byte[] data = new byte[SIZE];
for (int i = 0; i < SIZE; ++i) {
data[i] = (byte) (64 + (i % 32));
}
ByteArrayInputStream stream = new ByteArrayInputStream(data);
DriverManager.registerDriver(new OracleDriver());
Connection c = DriverManager.getConnection(
"jdbc:oracle:thin:@some_database", "user", "password");
String sql =
"DECLARE\n" +
" l_line CLOB;\n" +
"BEGIN\n" +
" l_line := ?;\n" +
" UPDATE table SET log = log || l_line || CHR(10) WHERE id = ?;\n" +
"END;\n";
PreparedStatement stmt = c.prepareStatement(sql);
stmt.setAsciiStream(1, stream, SIZE);
stmt.setInt(2, 1);
stmt.execute();
stmt.close();
c.commit();
c.close();
}
}
BLOB не изменяемы из SQL (ну, кроме того, что для них задано значение NULL), поэтому для добавления вам нужно сначала загрузить BLOB-объект, выполнить локальную конкатенацию и снова загрузить результат.
Обычное решение - записать несколько записей в базу данных с общим ключом и последовательностью, которая сообщает БД, как упорядочивать строки.