使用@Query和Hibernate在Spring Data JPA中更新布尔值

39
我已经配置并运行了spring-data和hibernate。我可以使用spring-data保存记录,但由于某些原因,我无法运行更新表中所有布尔字段的查询。
我尝试过这个:
@Query("update Content v set v.published = false where v.division = :division and v.section = :section")
void unPublishContent(@Param("division") String division, 
                         @Param("section") String section);
我也尝试了这个:
 @Query("update Content v set v.published = 0 where v.division = :division and v.section = :section")
void unPublishContent(@Param("division") String division, 
                         @Param("section") String section);

参数 division 和 section 已经生效,但表格没有改变。

p.s. 我还在使用 MySQL 数据库。

5个回答

116

我正在使用Spring 3.1和Spring JPA Data。我遇到了一个类似的问题,当我尝试在一个查询中更新多条记录时,经常会出现错误。

所以,我的代码大致是这样的。

@Query("UPDATE User u SET u.state = ?1 WHERE u.server.id = ?2")
public void updateAllUsers(long state, long serverid);

错误:

org.hibernate.hql.QueryExecutionRequestException: Not supported for DML operations

所以,经过一番谷歌搜索,我发现你需要添加@Modifying。

@Modifying  
@Query("UPDATE User u SET u.state = ?1 WHERE u.server.id = ?2")
public void updateAllUsers(long state, long serverid);

但是之后我遇到了以下错误:

...    
nested exception is org.springframework.dao.InvalidDataAccessApiUsageException: Executing an update/delete query; 
nested exception is javax.persistence.TransactionRequiredException: Executing an update/delete query
...

所以,我认为我的问题现在是一个事务问题,于是我回到谷歌上研究它,发现现在必须添加 @Transactional。看来 @Modifying 也需要 @Transactional。

@Modifying  
@Transactional
@Query("UPDATE User u SET u.state = ?1 WHERE u.server.id = ?2")
public void updateAllUsers(long state, long serverid);

但是我遇到了以下错误:

No value for key [org.apache.commons.dbcp.BasicDataSource (...) ] bound to thread

我又谷歌了一会儿,得出结论我的配置有误,这个结论证明是正确的。我缺少一些xml配置。

<beans:bean class="org.springframework.orm.jpa.JpaTransactionManager" id="transactionManager">
    <beans:property name="entityManagerFactory" ref="entityManagerFactory"/>
</beans:bean>

<tx:annotation-driven transaction-manager="transactionManager"/>

经历了漫长的旅程,但我终于让它运转起来了。希望这能帮助到某些人,像其他众多优秀博客、答案和评论帮助过我的一样,我也想“回馈社区”。


2
<tx:annotation-driven /> 不是必需的,因为Spring Data会自动将基于注释的事务应用于存储库接口。 - Oliver Drotbohm
1
请确保从org.springframework.transaction.annotation而不是javax导入注释。 - barryku
完美的答案!!非常感谢! - LETOURNEUR Léo

10

要执行修改查询,您需要按照参考文档中所概述的方式,在方法上添加额外的@Modifying注释,如下所示:

返回结果为:

要执行修改查询,您需要按照参考文档中所概述的方式,在方法上添加额外的@Modifying注释,如下所示:

@Modifying
@Query("update Content v set v.published = false where v.division = :division and v.section = :section")
void unPublishContent(@Param("division") String division, 
                     @Param("section") String section);

那个……这对你有用吗?我有点困惑你的意思 ;) - Oliver Drotbohm
1
抱歉 Oliver,我有点累了,打错了一个字。它并没有起到帮助作用。据我所知,如果您的存储库被注释为只读,则"@Modifying"选项很有用。这样,您可以拥有一个主要是只读的存储库,但仍允许您在其中编写方法。 - aki
这与readOnly无关 - 至少不是直接相关的。 readOnly涉及事务,而@Modifying涉及对EntityManager的调用。当然,在只读事务中尝试执行修改方法是没有意义的。 - Oliver Drotbohm

8

对于我来说,以下注释也起作用:

@Modifying
@Query("update User u set u.active=1 where a.id=?1")
@Transactional
void  activeUser(Long id);

4
对于我来说,以下注释有效:
@Modifying
@Query("update JsonContactImport x set x.isImported = true where x.id in :ids")
@Transactional
void updateImported(@Param("ids")  List<Long> ids);

0

在使用Spring 5时,仍然需要使用@Transactional注解。

@Repository
public interface DoorStyleRepository extends JpaRepository<DoorStyle, Long> {

    @Modifying
    @Transactional
    @Query("UPDATE DoorStyle ds SET ds.enabled = false")
    void setEnabledFalse();
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接