无法使用 Spring Boot 更新现有实体
Posted
技术标签:
【中文标题】无法使用 Spring Boot 更新现有实体【英文标题】:Existing entity cannot be updated with Spring Boot 【发布时间】:2016-06-08 09:07:49 【问题描述】:我的 Spring Boot 应用有以下类:
董事会(JPA 实体)
@Entity
@Table(name = "board")
public class Board
public static final int IN_PROGRESS = 1;
public static final int AFK = 2;
public static final int COMPLETED = 3;
@Column(name = "id")
@Generated(GenerationTime.INSERT)
@GeneratedValue(strategy = GenerationType.AUTO)
@Id
private Long id;
@Column(name = "status", nullable = false)
private int status = IN_PROGRESS;
BoardRepository(JPA 存储库)
public interface BoardRepository extends JpaRepository<Board, Long>
CommonBoardService(基础服务)
public interface CommonBoardService
Board save(Board board);
Board update(Board board, int status);
CommonBoardServiceImpl(基础服务实现)
@Service
@Transactional
public class CommonBoardServiceImpl implements CommonBoardService
@Autowired
private BoardRepository boardRepository;
public Board save(final Board board)
return boardRepository.save(board);
@Transactional(propagation = Propagation.REQUIRES_NEW)
public Board update(final Board board, final int status)
board.setStatus(status);
return save(board);
BoardService(具体服务接口)
public interface BoardService
Board startBoard();
void synchronizeBoardState(Board board);
BoardServiceImpl(具体服务实现)
@Service
@Transactional
public class BoardServiceImpl implements BoardService
@Autowired
private CommonBoardService commonBoardService;
public Board startBoard() return new Board();
public void synchronizeBoardState(final Board board)
if (board != null && inProgress(board))
if (!canPlayWithCurrentBoard(board))
commonBoardService.update(board, Board.AFK);
else
commonBoardService.update(board, Board.COMPLETED);
private boolean canPlayWithCurrentBoard(final Board board)
return !inProgress(board);
private boolean inProgress(final Board board)
return board != null && board.getStatus() == Board.IN_PROGRESS;
BoardServiceTest(单元测试)
1. @RunWith(SpringJUnit4ClassRunner.class)
2. @Transactional
3. public class BoardServiceTest
4. @Autowired
5. private BoardRepository boardRepository;
6.
7. @Autowired
8. private BoardService boardService;
9. @Autowired
10. private CommonBoardService commonBoardService;
11.
12. @Test
13. public void testSynchronizeBoardStatus()
14. Board board = boardService.startBoard();
15.
16. board = commonBoardService.save(board);
17.
18. assertEquals(1, boardRepository.count());
19.
20. boardService.synchronizeBoardState(board);
21.
22. assertEquals(1, boardRepository.count());
23.
24.
此测试在第 22 行失败,错误为 java.lang.AssertionError: Expected :1 Actual:2
。 Hibernate SQL 日志显示在第 20 行触发了 INSERT
,而不是 UPDATE
。由于我在整个测试过程中使用相同的 Board
对象,因此我希望第 20 行会触发 UPDATE
而不是 INSERT
。
谁能解释为什么会发生这种情况以及如何获得预期的行为(第 20 行的UPDATE
)?
【问题讨论】:
您的 TestDataSourceConfiguration.class 指向哪个数据库?它是否有您要更新的董事会实体?BoardServiceTest
中的boardService
和commonBoardService
是什么? boardService.startBoard(admin)
和 commonBoardService.save(board)
看起来像什么?大概,第一个是创建一个新的board
,第二个是保存它?另外,Board
和 User
类的代码是什么样的?如果您可以创建一个演示问题的小示例可能会更好,因为要涵盖的代码非常多。
@koder23,感谢您的回复。我正在使用postgres。是的,它应该有 Board board = boardService.startBoard(admin);
- 创建新的板,实体的 ID 等于 1。
你能贴出完整的代码来测试一下吗?仍有许多部分缺失,难以重现问题。可以在 Github 上发布示例应用吗?
@manish,你是对的,boardService.startBoard(admin)
- 为用户创建新版块。如您所见,boardService.synchronizeBoardState(board, admin)
有不同的检查。所以,我想在我们进入if (!canPlayWithCurrentBoard(board))
代码块时触发这个案例。为此,我需要更新board
实体以满足这些条件。这就是我在测试中使用commonBoardService.save(board)
的原因。
【参考方案1】:
罪魁祸首是这一行:@Transactional(propagation = Propagation.REQUIRES_NEW)
。让我们看看执行测试用例时会发生什么。
BoardServiceTest
用@Transactional
注释,所以当BoardServiceTest.testSynchronizeBoardStatus
开始执行时会启动一个新事务。
第 14 行创建一个新的 Board
实例。
第 16 行尝试保存在第 14 行创建的Board
实例并触发数据库INSERT
。
第 20 行间接调用 CommonBoardServiceImpl.update
,并用 @Transactional(propagation = Propagation.REQUIRES_NEW)
注释。这会暂停正在进行的事务(请参阅JavaDocs for Propagation),该事务目前既未提交也未回滚。
CommonBoardServiceImpl.update
反过来尝试保存传递给它的 Board
实例。
给定实例未被识别为现有实例,因为将其保存到数据库的事务当前处于挂起状态。因此,它被假定为一个新实例并产生第二个INSERT
。
第 20 行现在完成,提交为 CommonBoardServiceImpl.update
启动的内部事务。外部事务恢复。
第 22 行找到脏会话并在触发 SELECT
查询之前刷新它。这意味着数据库中现在有两个实例,因此测试失败。
删除@Transactional(propagation = Propagation.REQUIRES_NEW)
确保整个测试在同一个事务中执行并因此通过。
【讨论】:
以上是关于无法使用 Spring Boot 更新现有实体的主要内容,如果未能解决你的问题,请参考以下文章
当实体自引用时,Spring Boot 和 Hibernate 无法自动创建表
无法将 Spring Data MongoDB + Spring Data JPA 与 Spring Boot 一起使用
Spring Boot配置Mysql后无法根据java实体类生成table
Spring Boot 中的 Hibernate 无法懒惰地初始化角色集合,无法初始化代理 - 没有 Session 异常