如何从org.springframework.data.repository.CrudRepository后代中取消代理子对象

时间:2019-02-01 03:52:58

标签: java hibernate jpa spring-data-jpa lazy-loading

我已经在MySQL数据库中创建了用户与项目之间的多对多关系,并使用了hbm2java(用于从数据库表生成Java类的休眠工具)。由于我使用的是gradle插件org.hibernate.gradle.tools.Schema,所以我没有hibernate.cfg.xml。

我可以从CrudRepository<User,Long>的后代中获取并正确打印用户列表(请参见下面的代码),并且每个用户对象都有一个getProjects()函数。当我尝试迭代用户拥有的项目时,出现此错误:

org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: com.xyz.dbmodel.orm.User.projects, could not initialize proxy - no Session

我想坚持使用hbm2java生成的Java代码所使用的默认延迟加载方法。

我曾尝试致电Hibernate.Initialize(user.getProjects())(如https://howtodoinjava.com/hibernate/use-hibernate-initialize-to-initialize-proxycollection/所述),但尝试执行此操作仍会出现相同的错误

for(Project project : user.getProjects()) {... }

我可以找到的所有示例都假设我可以执行上述for循环并取消每个项目的代理。我不能。

这是我的存储库:

import org.springframework.data.repository.CrudRepository;
import com.xyz.dataservice.dbmodel.orm.User;
public interface UserRepository extends CrudRepository<User, Long> {}

在这里我创建存储库:

@Autowired com.xyz.repository.UserRepository userRepository;

我在这里成功获取了用户列表

Iterable<User> users = userRepository.findAll();
for(User user : users) {
   log.info("User="+user.getName()); // this works!
   Set<Project> projects = user.getProjects();
   for(Project p : projects) // Error happens here!
   {
      log.info("  project="+p.getName());
   }
 }

预期结果是我得到了每个用户的项目列表。 实际结果是org.hibernate.LazyInitializationException例外。

谢谢!

2019年2月2日上午更新

关于休眠配置

我不确定您的意思。我的gradle插件正在生成以下hibernate.cfg.xml文件:

 <?xml version="1.0" encoding="utf-8"?>
 <!DOCTYPE hibernate-configuration 
     SYSTEM "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
 <hibernate-configuration>
   <session-factory>
     <property name="hibernate.dialect">org.hibernate.dialect.MySQL5Dialect</property>
     <property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
     <property name="hibernate.connection.url">jdbc:mysql://localhost:3306</property>
     <property name="hibernate.connection.username">searchapp_user</property>
     <property name="hibernate.connection.password">secret</property>
     <property name="hibernate.current_session_context_class">thread</property>
     <property name="hibernate.connection.zeroDateTimeBehavior">convertToNull</property>
   </session-factory>
 </hibernate-configuration>

...

添加事务性

我已将org.springframework.transaction.annotation.Transactional添加到我正在调用的新函数中

@Transactional
public int fetchUsers() {
    Iterable<User> users = userRepository.findAll();
    return listUsers(users, "");
}

如果这样做没有帮助,我尝试使用Transactional属性增强存储库:

@Repository
public interface UserRepository extends CrudRepository<User, Long> {
  @Transactional
  public List<User> findAll();
}

糟糕,这也无济于事。 我确实在英亩的日志条目中注意到了这个奇怪的条目。这是什么意思?

 2019-02-01 10:50:42.285  INFO 5096 --- [  restartedMain] trationDelegate$BeanPostProcessorChecker : Bean 'org.springframework.transaction.annotation.ProxyTransactionManagementConfiguration' of type [org.springframework.transaction.annotation.ProxyTransactionManagementConfiguration$$EnhancerBySpringCGLIB$$6c9c1be1] is not eligible for getting processed by all BeanPostProcessors (for example: not eligible for auto-proxying)

再次感谢!

2019年2月2日下午更新

糟糕,我忘了检查与交易性文档相关的链接。

我不确定如何在同一函数上使用@Bean和@Transactional。

能给我举个例子吗?

我见过的@Bean示例返回一个lambda,然后稍后将其神秘地调用。我需要将lambda声明为事务性的吗?那可能吗?如果不是,我是否需要用java.util.function.Consumer <>的后代替换lambda,其后继函数使用@Transactional声明?

2019年2月5日更新:找到解决方案!

我正在使用新的Java8 lambda语法,因此不得不放弃它,转而使用较旧的内部类语法,以便可以使用@Transactional函数。

  @Transactional
  class InitSpecialCommandLineRunner implements org.springframework.boot.CommandLineRunner{
      @Transactional // this is important!
      @Override
      public void run(String[] args) {
          int count = listProjectsForEachUser(); // this works!
      }
  }


  @org.springframework.context.annotation.Profile("initialize")
  @org.springframework.context.annotation.Bean
  @Transactional
  public org.springframework.boot.CommandLineRunner initSpecial() {
       return new InitSpecialCommandLineRunner(); // this works

  // How to declare this transactional?
  //  return args ->
  //  {
  //      int count = fetchUsers();
  //      Iterable<User> users;
  //  };

}

我希望有一天能对其他人有所帮助。

1 个答案:

答案 0 :(得分:2)

看起来事务仅跨越对存储库的立即调用。

要解决此问题,请在应定义事务范围的方法上放置一个@Transactional注释。 至少托管您显示给我们代码的方法。

该方法必须是公共的,并且在Spring bean上。 有关详细信息,请参见https://docs.spring.io/spring/docs/4.2.x/spring-framework-reference/html/transaction.html