我已创建的用户的项目在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循环和unproxy每个项目的所有例子。我不能。
这里是我的仓库:
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
。
谢谢!
关于Hibernate配置
我不确定你是什么意思。我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, "");
}
当这并没有帮助,我试图加强与事务属性我的仓库:
@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)
再次感谢!
哎呀,我忘了检查有关事务的文档该链接。
我不知道如何使用@Bean和@Transactional上相同的功能。
你能告诉我一个例子?
我见过的例子@Bean返回拉姆达,然后将其称为神秘更高版本。我是否需要申报事务的拉姆达?那可能吗?如果不是,那我需要java.util.function.Consumer <>,其接受函数声明与@Transactional的后裔更换拉姆达?
我用的是新的Java 8 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;
// };
}
我希望这会帮助别人的一天。
它看起来像只交易跨越到资源库中直接调用。
为了解决这个问题把@Transactional
批注上应定义事务范围的方法。至少在托管代码的方法,你向我们展示。
这种方法需要在公众和在一个Spring bean。见https://docs.spring.io/spring/docs/4.2.x/spring-framework-reference/html/transaction.html了解详情。