我们有一个包含 100 多个实体类的广泛实体模型。所有实体类都是单个实体超类的子类。共享缓存模式已设置为ALL。
@Entity
@Inheritance(strategy = InheritanceType.JOINED)
@Table(name = "entities")
public abstract class LongIdEntity {
@Id
@GeneratedValue(strategy = GenerationType.TABLE)
private Long id;
@Version
@Column(name = "OPT_LOCK_VERSION")
private Long lockVersion;
etc...
}
一个示例子类:
@Entity
@Table(name = "cars")
public class Car extends LongIdEntity { ... }
我们想在二级缓存中缓存所有实体。问题是只为所有实体创建了 1 个缓存区域;命名为LongIdEntity。
调试显示 Hibernate确实找到了所有实体类,但无论如何都为它们分配了相同的区域。因为在SessionFactoryImpl:339:
String cacheRegionName = cacheRegionPrefix + model.getRootClass().getCacheRegionName();
在我们的例子中,对model.getRootClass()的调用总是会产生“ LongIdEntity ”。
我认为这确实会缓存所有实体,但没有任何驱逐控制。有些课程非常频繁且只读。因此,我们希望将它们固定在内存中。有些通常在特定的时间跨度内使用,等等......将它们全部塞进同一个缓存会使它全部失效。
在注释中指定区域无效。例如:
@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE,region = "cars")
@Entity
@Table(name = "cars")
public class Car extends LongIdEntity { ... }
奇怪的是,只有共享缓存模式ALL才能获取实体类。在任何其他模式下,没有实体 - 即使使用 @Cache 和/或 @Cacheable 注释。也许这是一个迹象?
有人知道我如何分配特定的实体类特定区域?
TIA :-)
persistence.xml是基本的:
<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.0" xmlns="http://java.sun.com/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd">
<persistence-unit name="cars" transaction-type="RESOURCE_LOCAL">
<shared-cache-mode>ALL</shared-cache-mode>
</persistence-unit>
</persistence>
会话工厂采用经典方式:
<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
<property name="persistenceProviderClass" value="org.hibernate.ejb.HibernatePersistence"/>
<property name="persistenceUnitName" value="optimus"/>
<property name="dataSource" ref="dataSource"/>
<property name="jpaProperties">
<props>
<prop key="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.SingletonEhCacheRegionFactory</prop>
<prop key="hibernate.cache.use_second_level_cache">true</prop>
<prop key="hibernate.cache.use_query_cache">true</prop>
<prop key="hibernate.generate_statistics">true</prop>
<prop key="hibernate.cache.default_cache_concurrency_strategy">NONSTRICT_READ_WRITE</prop>
<prop key="hibernate.hbm2ddl.auto">update</prop>
<prop key="hibernate.jdbc.batch_size">1000</prop>
<prop key="hibernate.show_sql">false</prop>
<prop key="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</prop>
<prop key="hibernate.search.default.directory_provider">filesystem</prop>
<prop key="hibernate.search.default.indexBase">/hibernate-search</prop>
</props>
</property>
</bean>
环境
- JDK6
- Linux x64
- 休眠 4.1.10
- 春天 3.2.1
更新:使用@MappedSuperclass
@MappedSuperclass
@Inheritance(strategy = InheritanceType.JOINED)
public abstract class LongIdEntity { ... }
也不改变一件事。