休眠与缓存的连接过多
Hibernate Too Many Connections With Cache
我正在使用休眠来尝试检索缓存的查询。
@Transactional
public interface ProductDAO extends JpaRepository<Product, Long> {
@QueryHints({ @QueryHint(name = "org.hibernate.cacheable", value = "true") })
Product findByCode(String code);
}
我正在进行负载测试,我正在 1000 次迭代的大循环中执行此操作。
for (int i = 0; i < 500; i++) {
URL myURL = new URL("http://localhost:8080/test");
URLConnection myURLConnection = myURL.openConnection();
myURLConnection.connect();
myURLConnection.getContent();
}
我已经用 showsql 进行了检查,我发现只有 1 个 SQL 语句是为我第一次访问数据库而生成的,之后它被缓存了。
但即使没有显示 SQL,我仍然收到以下错误:
com.mysql.jdbc.exceptions.jdbc4.MySQLNonTransientConnectionException: Too many connections
我的 Hibernate 属性:
#hibernate properties
hibernate.dialect = ${hibernate.dialect}
hibernate.show_sql = false
hibernate.hbm2ddl.auto = ${hibernate.hbm2ddl}
hibernate.c3p0.min_size = 10
hibernate.c3p0.max_size = 100
hibernate.c3p0.timeout = 300
hibernate.c3p0.max_statements = 50
hibernate.c3p0.acquire_increment = 5
hibernate.c3p0.idle_test_period = 3000
hibernate.cache.use_second_level_cache=true
hibernate.cache.region.factory_class=org.hibernate.cache.ehcache.SingletonEhCacheRegionFactory
hibernate.cache.use_query_cache=true
数据库配置:
<bean name="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="url" value="${db.url}" />
<property name="driverClassName" value="${db.driverClassName}" />
<property name="username" value="${db.username}" />
<property name="password" value="${db.password}" />
</bean>
<bean class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean" id="entityManagerFactory">
<property name="dataSource" ref="dataSource" />
<property name="jpaVendorAdapter">
<bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
<property name="showSql" value="false" />
<property name="generateDdl" value="true" />
<property name="databasePlatform" value="${hibernate.dialect}" />
</bean>
</property>
<property name="jpaProperties" ref="hibernateProperties" />
<property name="packagesToScan">
<array>
<value>com.exammple.model</value>
</array>
</property>
</bean>
<bean id="hibernateProperties" class="org.springframework.beans.factory.config.PropertiesFactoryBean">
<property name="location" value="classpath:/spring/hibernate.properties" />
</bean>
<bean id="sessionFactory" factory-bean="entityManagerFactory" factory-method="getSessionFactory" />
<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
<property name="dataSource" ref="dataSource" />
<property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>
<tx:annotation-driven transaction-manager="transactionManager" />
<jpa:repositories base-package="com.example.dal" entity-manager-factory-ref="entityManagerFactory"
transaction-manager-ref="transactionManager" repository-impl-postfix="CustomImpl" />
这个问题可能有多个问题:
您没有关闭数据库连接。
您设置的最大 connection pool 池大小超过了数据库服务器允许的最大连接数。当客户端数量超过允许的最大连接数时,你会得到这样的异常。
我可以使用 FlexyPool 来监视连接池的使用情况,并查明连接是否泄漏或是否被长期租用。
更新
正如 M. Deinum 所说,您没有使用连接池。
您仍然可以使用 Hibernate C3P0 属性,但您必须从 LocalContainerEntityManagerFactoryBean
:
中删除 dataSource
<property name="dataSource" ref="dataSource" />
这样 Hibernate 就可以使用 hibernate.c3p0
属性。
问题是您的配置没有使用连接池。
您配置的 DriverManagerDataSource
不是正确的连接池。您正在将此 bean 注入 LocalContainerEntityManagerFactoryBean
,这会使您的 hibernate.connection
和 hibernate.c3p0
属性无用,它们未被使用。
解决方案非常简单,删除 hibernate.c3p0
和 hibernate.connection
属性并用适当的池实现替换 DriverManagerDataSource
。我会推荐 HikariCP 而不是 C3P0,但这是个人喜好。
<bean id="hikariConfig" class="com.zaxxer.hikari.HikariConfig">
<property name="poolName" value="springHikariCP" />
<property name="connectionTestQuery" value="SELECT 1" />
<property name="dataSourceClassName" value="${db.driverClassName}" />
<property name="dataSourceProperties">
<props>
<prop key="url">${db.url}</prop>
<prop key="user">${db.username}</prop>
<prop key="password">${jdb.password}</prop>
</props>
</property>
</bean>
<bean id="dataSource" class="com.zaxxer.hikari.HikariDataSource" destroy-method="close">
<constructor-arg ref="hikariConfig" />
</bean>
我正在使用休眠来尝试检索缓存的查询。
@Transactional
public interface ProductDAO extends JpaRepository<Product, Long> {
@QueryHints({ @QueryHint(name = "org.hibernate.cacheable", value = "true") })
Product findByCode(String code);
}
我正在进行负载测试,我正在 1000 次迭代的大循环中执行此操作。
for (int i = 0; i < 500; i++) {
URL myURL = new URL("http://localhost:8080/test");
URLConnection myURLConnection = myURL.openConnection();
myURLConnection.connect();
myURLConnection.getContent();
}
我已经用 showsql 进行了检查,我发现只有 1 个 SQL 语句是为我第一次访问数据库而生成的,之后它被缓存了。
但即使没有显示 SQL,我仍然收到以下错误:
com.mysql.jdbc.exceptions.jdbc4.MySQLNonTransientConnectionException: Too many connections
我的 Hibernate 属性:
#hibernate properties
hibernate.dialect = ${hibernate.dialect}
hibernate.show_sql = false
hibernate.hbm2ddl.auto = ${hibernate.hbm2ddl}
hibernate.c3p0.min_size = 10
hibernate.c3p0.max_size = 100
hibernate.c3p0.timeout = 300
hibernate.c3p0.max_statements = 50
hibernate.c3p0.acquire_increment = 5
hibernate.c3p0.idle_test_period = 3000
hibernate.cache.use_second_level_cache=true
hibernate.cache.region.factory_class=org.hibernate.cache.ehcache.SingletonEhCacheRegionFactory
hibernate.cache.use_query_cache=true
数据库配置:
<bean name="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="url" value="${db.url}" />
<property name="driverClassName" value="${db.driverClassName}" />
<property name="username" value="${db.username}" />
<property name="password" value="${db.password}" />
</bean>
<bean class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean" id="entityManagerFactory">
<property name="dataSource" ref="dataSource" />
<property name="jpaVendorAdapter">
<bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
<property name="showSql" value="false" />
<property name="generateDdl" value="true" />
<property name="databasePlatform" value="${hibernate.dialect}" />
</bean>
</property>
<property name="jpaProperties" ref="hibernateProperties" />
<property name="packagesToScan">
<array>
<value>com.exammple.model</value>
</array>
</property>
</bean>
<bean id="hibernateProperties" class="org.springframework.beans.factory.config.PropertiesFactoryBean">
<property name="location" value="classpath:/spring/hibernate.properties" />
</bean>
<bean id="sessionFactory" factory-bean="entityManagerFactory" factory-method="getSessionFactory" />
<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
<property name="dataSource" ref="dataSource" />
<property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>
<tx:annotation-driven transaction-manager="transactionManager" />
<jpa:repositories base-package="com.example.dal" entity-manager-factory-ref="entityManagerFactory"
transaction-manager-ref="transactionManager" repository-impl-postfix="CustomImpl" />
这个问题可能有多个问题:
您没有关闭数据库连接。
您设置的最大 connection pool 池大小超过了数据库服务器允许的最大连接数。当客户端数量超过允许的最大连接数时,你会得到这样的异常。
我可以使用 FlexyPool 来监视连接池的使用情况,并查明连接是否泄漏或是否被长期租用。
更新
正如 M. Deinum 所说,您没有使用连接池。
您仍然可以使用 Hibernate C3P0 属性,但您必须从 LocalContainerEntityManagerFactoryBean
:
dataSource
<property name="dataSource" ref="dataSource" />
这样 Hibernate 就可以使用 hibernate.c3p0
属性。
问题是您的配置没有使用连接池。
您配置的 DriverManagerDataSource
不是正确的连接池。您正在将此 bean 注入 LocalContainerEntityManagerFactoryBean
,这会使您的 hibernate.connection
和 hibernate.c3p0
属性无用,它们未被使用。
解决方案非常简单,删除 hibernate.c3p0
和 hibernate.connection
属性并用适当的池实现替换 DriverManagerDataSource
。我会推荐 HikariCP 而不是 C3P0,但这是个人喜好。
<bean id="hikariConfig" class="com.zaxxer.hikari.HikariConfig">
<property name="poolName" value="springHikariCP" />
<property name="connectionTestQuery" value="SELECT 1" />
<property name="dataSourceClassName" value="${db.driverClassName}" />
<property name="dataSourceProperties">
<props>
<prop key="url">${db.url}</prop>
<prop key="user">${db.username}</prop>
<prop key="password">${jdb.password}</prop>
</props>
</property>
</bean>
<bean id="dataSource" class="com.zaxxer.hikari.HikariDataSource" destroy-method="close">
<constructor-arg ref="hikariConfig" />
</bean>