使用CLOB绑定通过JDBC执行sql语句

时间:2009-03-24 15:55:51

标签: oracle jdbc clob sqlbindparameter ora-01461

我有以下查询(列日志的类型为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 个答案:

答案 0 :(得分:2)

BLOB在SQL中是不可变的(好吧,除了将它们设置为NULL),所以要附加,你必须首先下载blob,在本地连接,然后再次上传结果。

通常的解决方案是使用公共密钥和序列向数据库写入多条记录,该序列告诉DB如何对行进行排序。

答案 1 :(得分: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();
    }
}