21

最近迁移到 POSTGRESQL,我试图在创建 db 表的新条目时获取唯一生成的密钥。该表screenstable如下所示:

CREATE TABLE screenstable
(
  id serial NOT NULL,
  screenshot bytea,
  CONSTRAINT screen_id PRIMARY KEY (id )
)

插入数据的方法screenstable如下:

@Autowired NamedParameterJDBCTemplate template;
public int insertImage(ImageBean imageBean){
        String query = "insert into screenstable (screenshot) values (:image)";
        SqlParameterSource data = new BeanPropertySqlParameterSource(imageBean);
        KeyHolder keyHolder = new GeneratedKeyHolder();
        template.update(query, data, keyHolder);
        return keyHolder.getKey().intValue();
    }

并且ImageBean

import java.util.Arrays;

public class ImageBean {
    private int id;
    private byte[] image;
    @Override
    public String toString() {
        return "ImageBean [id=" + id + ", image=" + Arrays.toString(image)
                + "]";
    }
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public byte[] getImage() {
        return image;
    }
    public void setImage(byte[] image) {
        this.image = image;
    }
}

但是运行代码会出现以下异常

15:33:20,953 ERROR JsonParseExceptionMapper:15 - org.springframework.dao.InvalidDataAccessApiUsageException: The getKey method should only be used when a single key is returned.  The current key entry contains multiple keys: [{id=3, screenshot=[B@db59df}]
org.springframework.dao.InvalidDataAccessApiUsageException: The getKey method should only be used when a single key is returned.  The current key entry contains multiple keys: [{id=3, screenshot=[B@db59df}]
        at org.springframework.jdbc.support.GeneratedKeyHolder.getKey(GeneratedKeyHolder.java:65)
        at some.project.model.FeedbackDao.insertImage(FeedbackDao.java:20)
        at some.project.rest.FeedsRest.pluginCheck(FeedsRest.java:62)
        at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
        at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
        at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
        at java.lang.reflect.Method.invoke(Method.java:597).....

与 POSTGRES 一起使用时,相同的代码可以正常运行,MySQL但在使用键时会失败。数据类型是否以serial某种方式导致代码失败,或者我是否正确使用了主键功能?

请指教。

4

3 回答 3

27

如果框架没有被告知哪一列是键,它将返回表的所有列作为键。

您可以通过将新参数传递给更新方法来通知它,如下所示:

template.update(query, data, keyHolder, new String[] { "id" });

请参见NamedParameterJdbcTemplate.update(sql, paramSource, generatedKeyHolder, keyColumnNames)

于 2014-03-19T22:42:06.260 回答
5

您也可以使用 JdbcTemplate 但在这种情况下您必须检查它:

    jdbcTemplate.update(this, holder);
    Long newId;
    if (holder.getKeys().size() > 1) {
        newId = (Long)holder.getKeys().get("your_id_column");
    } else {
        newId= holder.getKey().longValue();
    }

GeneratedKeyHolder类抛出异常,因为类只知道当有一个键时应该返回什么(并且该键需要是 Number 实例,因为 getKey 方法返回 Number 对象)。

在 jdbc 驱动程序返回多个键的情况下,您必须确定生成的键列(更新之前或之后)。请查看 GeneratedKeyHolder#getKey 的源代码 - 分析起来非常简单。检查键列表大小并且超过键 Spring 不知道应该返回哪个键,这就是返回异常的原因。

注意:请记住,该方法不适用于 Oracle,因为 Oracle 返回类似 ROWID 的内容。对于 Oracle,您必须使用 NamedParameterJdbcTemplate。

于 2015-09-02T19:31:43.640 回答
0

只是快速浏览的猜测...... keyHolder.getKey().intValue() 应该是 keyHolder.getKey()["id"].intValue() (或类似的东西)?getKey() 函数返回一个具有多个键(id 和图像)的对象,并且您正在尝试使用 intValue() 函数将对象转换为 int,除非您指向从返回的对象中的“id”键,否则我猜这不起作用“getKey()” 干杯,弗朗西斯科

于 2013-07-21T12:05:07.820 回答