使用Spring JDBC在事务中插入新的父子记录时出错

时间:2010-11-26 13:54:13

标签: spring transactions spring-jdbc spring-transactions

我有2个表(项目和代码),关系一对多(一个项目可以有很多代码 - 这是在带有外键的数据库级别上实现的)。

当我尝试使用spring JDBC在一个事务中插入一条Item记录和几条Code记录(它们链接到该Item记录)时,我得到了

com.mysql.jdbc.exceptions.jdbc4.MySQLIntegrityConstraintViolationException: Cannot add or update a child row: a foreign key constraint fails (`mrbcodes`.`code`, CONSTRAINT `code_item_fk` FOREIGN KEY (`itemId`) REFERENCES `item` (`id`) ON DELETE NO ACTION ON UPDATE NO ACTION)

这是我插入数据库代码。

public void saveCode(Code code, long itemId) {
  String insertSql = "insert into code(number, value, itemId) values (:number,:value,:itemId)";
  SqlParameterSource namedParameters = new BeanPropertySqlParameterSource(code);
  log.info("Inserting values into code table; number:" + code.getNumber() +"; value:" + code.getValue() 
    + "; itemId:" + itemId);
  simpleJdbcTemplate.update(insertSql, namedParameters);
 }

 @Transactional
 public void saveCodes(Collection <Code> codes) {
  KeyHolder keyHolder = new GeneratedKeyHolder();
  insertNewItem(keyHolder);  
  Number key = keyHolder.getKey();
  Assert.notNull(key, "After item insert; returned key must not be null");
  for (Code code:codes) {
   saveCode(code, key.longValue());
  }
 }

 private KeyHolder insertNewItem(KeyHolder keyHolder) {
  String createItemSQL = "insert into item(timeUsed) values(:defaultValue)";  
  SqlParameterSource sqlParameterSource = new MapSqlParameterSource("defaultValue", defaultValue);
  log.debug("Inserting new item");
  simpleJdbcTemplate.getNamedParameterJdbcOperations().update(createItemSQL,sqlParameterSource, keyHolder);
  log.debug("Item inserted; keyHolder " + keyHolder);
  return keyHolder;
 }

1 个答案:

答案 0 :(得分:3)

您不会在itemId中使用saveCode,因此Code会在没有itemId的情况下保存。