Spring JDBC:即使将行插入表中,也不返回自动生成的ID
Posted
技术标签:
【中文标题】Spring JDBC:即使将行插入表中,也不返回自动生成的ID【英文标题】:Spring JDBC: Auto generated id is not returned even though the row is inserted in table 【发布时间】:2018-11-10 06:36:52 【问题描述】:我正在尝试在表 Taco
中插入行并获取自动生成的 ID。
当我尝试在saveTacoInfo
方法中使用keyHolder.getKey().longValue()
获取密钥时,它会抛出NPE,但是当我从H2-console 检查它时,它会显示插入到Taco
表中的记录。
我正在使用 Spring Boot 2.1.0、Spring 5.1.2 和嵌入式 H2 数据库。我该如何解决这个问题?
H2 表架构:
create table if not exists Taco (
id identity,
name varchar(50) not null,
createdAt timestamp not null
);
Jdbc 存储库实现:
package tacos.data;
import java.sql.Timestamp;
import java.sql.Types;
import java.util.Arrays;
import java.util.Date;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.core.PreparedStatementCreator;
import org.springframework.jdbc.core.PreparedStatementCreatorFactory;
import org.springframework.jdbc.support.GeneratedKeyHolder;
import org.springframework.jdbc.support.KeyHolder;
import org.springframework.stereotype.Repository;
import lombok.extern.slf4j.Slf4j;
import tacos.Taco;
@Slf4j
@Repository
public class JdbcTacoRepository implements TacoRepository
private JdbcTemplate jdbc;
public JdbcTacoRepository(JdbcTemplate jdbc)
this.jdbc = jdbc;
@Override
public Taco save(Taco taco)
long tacoId = saveTacoInfo(taco);
taco.setId(tacoId);
for (String ingredient : taco.getIngredients())
saveIngredientToTaco(ingredient, tacoId);
return taco;
private void saveIngredientToTaco(String ingredient, long tacoId)
jdbc.update("insert into Taco_Ingredients (taco, ingredient) values (?, ?)",
tacoId, ingredient);
private long saveTacoInfo(Taco taco)
taco.setCreatedAt(new Date());
log.info("taco: " + taco);
PreparedStatementCreator psc = new PreparedStatementCreatorFactory(
"insert into Taco (name, createdAt) values (?, ?)",
Types.VARCHAR, Types.TIMESTAMP
).newPreparedStatementCreator(
Arrays.asList(taco.getName(), new Timestamp(taco.getCreatedAt().getTime()))
);
KeyHolder keyHolder = new GeneratedKeyHolder();
jdbc.update(psc, keyHolder);
log.info("keyholder: " + keyHolder.getKeyList());
return keyHolder.getKey().longValue();
堆栈跟踪:
2018-11-10 11:18:34.459 INFO 4024 --- [nio-8080-exec-3] tacos.data.JdbcTacoRepository : taco: Taco(id=null, createdAt=Sat Nov 10 11:18:34 IST 2018, name=ccccccc, ingredients=[GRBF, CARN])
2018-11-10 11:18:34.542 INFO 4024 --- [nio-8080-exec-3] tacos.data.JdbcTacoRepository : keyholder: []
2018-11-10 11:18:34.597 ERROR 4024 --- [nio-8080-exec-3] o.a.c.c.C.[.[.[/].[dispatcherServlet] : Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processing failed; nested exception is java.lang.NullPointerException] with root cause
java.lang.NullPointerException: null
at tacos.data.JdbcTacoRepository.saveTacoInfo(JdbcTacoRepository.java:63) ~[classes/:na]
at tacos.data.JdbcTacoRepository.save(JdbcTacoRepository.java:31) ~[classes/:na]
at tacos.data.JdbcTacoRepository$$FastClassBySpringCGLIB$$59d8a28e.invoke(<generated>) ~[classes/:na]
at org.springframework.cglib.proxy.MethodProxy.invoke(MethodProxy.java:218) ~[spring-core-5.1.2.RELEASE.jar:5.1.2.RELEASE]
at org.springframework.aop.framework.CglibAopProxy$CglibMethodInvocation.invokeJoinpoint(CglibAopProxy.java:746) ~[spring-aop-5.1.2.RELEASE.jar:5.1.2.RELEASE]
at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:163) ~[spring-aop-5.1.2.RELEASE.jar:5.1.2.RELEASE]
at org.springframework.dao.support.PersistenceExceptionTranslationInterceptor.invoke(PersistenceExceptionTranslationInterceptor.java:139) ~[spring-tx-5.1.2.RELEASE.jar:5.1.2.RELEASE]
at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:186) ~[spring-aop-5.1.2.RELEASE.jar:5.1.2.RELEASE]
at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.intercept(CglibAopProxy.java:688) ~[spring-aop-5.1.2.RELEASE.jar:5.1.2.RELEASE]
at tacos.data.JdbcTacoRepository$$EnhancerBySpringCGLIB$$996b212f.save(<generated>) ~[classes/:na]
at tacos.web.DesignTacoController.processDesign(DesignTacoController.java:85) ~[classes/:na]
【问题讨论】:
【参考方案1】:我刚刚遇到了完全相同的问题。
解决方法很简单:
h2 版本必须是 1.4.196,因为 1.4.197 会导致这个问题。 只需在 pom.xml 中为 h2 依赖项指定此版本即可:
<dependency>
<groupId>com.h2database</groupId>
<artifactId>h2</artifactId>
<scope>runtime</scope>
**<version>1.4.196</version>**
</dependency>
还将 ``parent``` 设置为 2.0.4.RELEASE 版本。这个版本可以解决这个问题。
保存 pom.xml 并检查 maven 依赖项;确保h2和spring boot的版本已经相应改变。
我相信这可以解决您的问题。
【讨论】:
只需添加 H2 版本就可以了。您无需更改您的 Spring Boot 版本。另外,参考问题:github.com/h2database/h2database/issues/1052【参考方案2】:我今天遇到了同样的问题。
将 1.4.196 添加到 com.h2database 依赖项中有效。
【讨论】:
以上是关于Spring JDBC:即使将行插入表中,也不返回自动生成的ID的主要内容,如果未能解决你的问题,请参考以下文章