jjzjj

java - Hibernate 不会加载一对多的关系集,即使使用 eager fetch

coder 2024-03-20 原文

我有以下 hibernate 映射。

@OneToMany(fetch = FetchType.EAGER, cascade = CascadeType.ALL, mappedBy = "product")
private Set<ProductLicense> productLicenses = new HashSet<ProductLicense>(0);


@ManyToOne(fetch = FetchType.EAGER)
@JoinColumn(name = "product_id", nullable = false)
private Product product;

但是当我调用 product.getProductLicences() 时,我总是得到一个空的 Set,即使在事务方法中也是如此。

sessionFactory.getCurrentSession().get(Product.class, productId))
            .getProductLicenses()

下面是ProductDaoImpl类

@Repository
public class ProductDaoImpl implments ProductDao{

@Autowired
private SessionFactory sessionFactory;

@Override
public Product getProduct(Integer productId)
{

    Product result = (Product) sessionFactory.getCurrentSession().get(Product.class, productId);
    Hibernate.initialize(result.getProductLicenses());
            //sessionFactory.getCurrentSession().refresh(result);
    return result;

}
   .. other methods

}

此外,如果我调用 Hibernate.initialize(product);不执行表之间的任何连接。 指令sessionFactory.getCurrentSession().get(Product.class, productId);不会对数据库产生任何查询(我的属性 show_sql 等于 true)。 但是如果我取消注释 sessionFactory.getCurrentSession().refresh(result);我可以看到 sql 和集合已加载,但我不明白为什么在另一种情况下未加载。 但是我不明白我的映射有什么问题。

产品类别是:

@Entity
@Table(name = "product")
public class Product {

@Id
@GeneratedValue(strategy = IDENTITY)
@Column(name = "product_id", unique = true)
private Integer productId;

@NotNull
@Size(max = 200)
@Column(name = "name")
private String name;

    @OneToMany(fetch = FetchType.EAGER, cascade = CascadeType.ALL, mappedBy = "product")
private Set<ProductLicense> productLicenses = new HashSet<ProductLicense>(0);


    public Set<ProductLicense> getProductLicenses()
{
    return productLicenses;
}

public void setProductLicenses(Set<ProductLicense> productLicenses)
{
    this.productLicenses = productLicenses;
}

//getters and setters
..

}

ProductLicense 类:

@Entity
@Table(name = "product_license")
public class ProductLicense {

@Id
@GeneratedValue(strategy = IDENTITY)
@Column(name = "product_license_id", unique = true)
private Integer productLicenseId;

@ManyToOne(fetch = FetchType.EAGER)
@JoinColumn(name = "product_id", nullable = false)
    private Product product;

@NotNull
@Column(name = "key")
private String key;

    // getters and setters ...

}

最后我的配置如下:

 @Configuration
 @EnableWebMvc
 @EnableTransactionManagement
 @ComponentScan("com.company.package")
 @PropertySource("classpath:application.properties")

public class WebAppConfig {

private static final String PROPERTY_NAME_DATABASE_DRIVER = "db.driver";
private static final String PROPERTY_NAME_DATABASE_PASSWORD = "db.password";
private static final String PROPERTY_NAME_DATABASE_URL = "db.url";
private static final String PROPERTY_NAME_DATABASE_USERNAME = "db.username";

private static final String PROPERTY_NAME_HIBERNATE_DIALECT = "hibernate.dialect";
private static final String PROPERTY_NAME_HIBERNATE_SHOW_SQL = "hibernate.show_sql";
private static final String PROPERTY_NAME_ENTITYMANAGER_PACKAGES_TO_SCAN = "entitymanager.packages.to.scan";

@Resource
private Environment env;

@Bean
public DataSource dataSource()
{
    DriverManagerDataSource dataSource = new DriverManagerDataSource();

    dataSource.setDriverClassName(env.getRequiredProperty(PROPERTY_NAME_DATABASE_DRIVER));
    dataSource.setUrl(env.getRequiredProperty(PROPERTY_NAME_DATABASE_URL));
    dataSource.setUsername(env.getRequiredProperty(PROPERTY_NAME_DATABASE_USERNAME));
    dataSource.setPassword(env.getRequiredProperty(PROPERTY_NAME_DATABASE_PASSWORD));

    return dataSource;
}

private Properties hibProperties()
{
    Properties properties = new Properties();
    properties.put(PROPERTY_NAME_HIBERNATE_DIALECT,
            env.getRequiredProperty(PROPERTY_NAME_HIBERNATE_DIALECT));
    properties.put(PROPERTY_NAME_HIBERNATE_SHOW_SQL,
            env.getRequiredProperty(PROPERTY_NAME_HIBERNATE_SHOW_SQL));
    properties.put("hibernate.cache.provider_class", "org.hibernate.cache.NoCacheProvider");
    properties.put("hibernate.cache.use_second_level_cache", "false");
    return properties;
}

@Bean
public LocalSessionFactoryBean sessionFactory()
{
    LocalSessionFactoryBean lsfb = new LocalSessionFactoryBean();
    lsfb.setDataSource(this.dataSource());
    lsfb.setPackagesToScan(env.getRequiredProperty(PROPERTY_NAME_ENTITYMANAGER_PACKAGES_TO_SCAN));
    lsfb.setHibernateProperties(this.hibProperties());
    return lsfb;
}

@Bean
public HibernateTransactionManager transactionManager()
{
    return new HibernateTransactionManager(this.sessionFactory().getObject());
}
 }

我的测试:

@Before
@Transactional
public void setUp() throws ParseException
{

    product1 = new Product();
    ..

    productService.addProduct(product1);

    product2 = new Product();
    ..

    product2 = productService.addProduct(product2);

    productService.addProductLicense(product2.getProductId(), licenceKey1Product2);
    productService.addProductLicense(product2.getProductId(), licenceKey2Product2);



}

  @Test
@Transactional
public void addProductLicenseTest()
{
    String licenseKey[] = { "thisisthekey", "thisisthekey2" };

    productService.addProductLicense(product1.getProductId(), licenseKey[0]);

            //sessionFactory.getCurrentSession().flush();
    //sessionFactory.getCurrentSession().clear();

    Product product1saved = productService.getProduct(product1.getProductId());

    assertEquals(1, product1saved.getProductLicenses().size());
    assertEquals(licenseKey[0], product1saved.getProductLicenses().iterator().next().getKey());

    productService.addProductLicense(product1.getProductId(), licenseKey[1]);

    product1saved = productService.getProduct(product1saved.getProductId());

    assertEquals(2, product1saved.getProductLicenses().size());

 }

最佳答案

这是(很可能)发生的事情。

您的整个测试方法都是事务性的。因此,服务调用与测试代码本身发生在同一事务中。

在测试中,您将在没有任何许可的情况下创建并保留一个产品。这将 Product 实例存储在 session 缓存中,与事务相关联。该产品的许可证列表为空。

然后您仍然在同一个事务中调用一个服务方法,该方法创建一个许可证,其产品是您之前创建的产品。我的猜测是此服务的代码如下所示:

Product product = (Product) session.get(Product.class, productId);
ProductLicense license = new ProductLicense();
license.setProduct(product);
session.persist(license);

在上面的代码中,产品是从 session 缓存中获取的,然后许可证与找到的产品一起保存。请注意,代码设置了许可证的产品,但没有将许可证添加到产品中,这就是问题所在。

然后您将通过 ID 获取产品,仍在同一笔交易中。因此,Hibernate 直接从缓存中检索产品。而在缓存中,由于您省略了向产品添加许可证,因此许可证列表仍然是空的。

因此,简而言之,您有责任维护双向关联的双方。如果您只初始化所有者端,Hibernate 将保留关联,并且如果它从数据库加载产品,将正确初始化列表。但是,当产品在缓存中时,它会保持在您将其存储在缓存中的状态。这就是您看到空列表的原因,除非您明确清除 session 并因此从数据库重新加载状态。

关于java - Hibernate 不会加载一对多的关系集,即使使用 eager fetch,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22011757/

有关java - Hibernate 不会加载一对多的关系集,即使使用 eager fetch的更多相关文章

  1. ruby - Highline 询问方法不会使用同一行 - 2

    设置:狂欢ruby1.9.2高线(1.6.13)描述:我已经相当习惯在其他一些项目中使用highline,但已经有几个月没有使用它了。现在,在Ruby1.9.2上全新安装时,它似乎不允许在同一行回答提示。所以以前我会看到类似的东西:require"highline/import"ask"Whatisyourfavoritecolor?"并得到:Whatisyourfavoritecolor?|现在我看到类似的东西:Whatisyourfavoritecolor?|竖线(|)符号是我的终端光标。知道为什么会发生这种变化吗? 最佳答案

  2. ruby - 如何在续集中重新加载表模式? - 2

    鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

  3. ruby-on-rails - 项目升级后 Pow 不会更改 ruby​​ 版本 - 2

    我在我的Rails项目中使用Pow和powifygem。现在我尝试升级我的ruby​​版本(从1.9.3到2.0.0,我使用RVM)当我切换ruby​​版本、安装所有gem依赖项时,我通过运行railss并访问localhost:3000确保该应用程序正常运行以前,我通过使用pow访问http://my_app.dev来浏览我的应用程序。升级后,由于错误Bundler::RubyVersionMismatch:YourRubyversionis1.9.3,butyourGemfilespecified2.0.0,此url不起作用我尝试过的:重新创建pow应用程序重启pow服务器更新战俘

  4. java - 等价于 Java 中的 Ruby Hash - 2

    我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

  5. ruby - RuntimeError(自动加载常量 Apps 多线程时检测到循环依赖 - 2

    我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("

  6. ruby - 即使失败也继续进行多主机测试 - 2

    我已经构建了一些serverspec代码来在多个主机上运行一组测试。问题是当任何测试失败时,测试会在当前主机停止。即使测试失败,我也希望它继续在所有主机上运行。Rakefile:namespace:specdotask:all=>hosts.map{|h|'spec:'+h.split('.')[0]}hosts.eachdo|host|begindesc"Runserverspecto#{host}"RSpec::Core::RakeTask.new(host)do|t|ENV['TARGET_HOST']=hostt.pattern="spec/cfengine3/*_spec.r

  7. ruby-on-rails - 使用 config.threadsafe 时从 lib/加载模块/类的正确方法是什么!选项? - 2

    我一直致力于让我们的Rails2.3.8应用程序在JRuby下正确运行。一切正常,直到我启用config.threadsafe!以实现JRuby提供的并发性。这导致lib/中的模块和类不再自动加载。使用config.threadsafe!启用:$rubyscript/runner-eproduction'pSim::Sim200Provisioner'/Users/amchale/.rvm/gems/jruby-1.5.1@web-services/gems/activesupport-2.3.8/lib/active_support/dependencies.rb:105:in`co

  8. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  9. ruby - Rails 关联 - 同一个类的多个 has_one 关系 - 2

    我的问题的一个例子是体育游戏。一场体育比赛有两支球队,一支主队和一支客队。我的事件记录模型如下:classTeam"Team"has_one:away_team,:class_name=>"Team"end我希望能够通过游戏访问一个团队,例如:Game.find(1).home_team但我收到一个单元化常量错误:Game::team。谁能告诉我我做错了什么?谢谢, 最佳答案 如果Gamehas_one:team那么Rails假设您的teams表有一个game_id列。不过,您想要的是games表有一个team_id列,在这种情况下

  10. java - 我的模型类或其他类中应该有逻辑吗 - 2

    我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我

随机推荐