Hibernate, C3P0, Mysql - Сломанная труба
MySQL, похоже, имеет 8-часовой тайм-аут на своих подключениях. Я запускаю несколько WARs в Tomcat, используя Hibernate для ORM. Через 8 часов (т.е. За одну ночь) я получаю разбитые трубы, когда он поднимает простоя.
Я уже отслеживал код и делал вдвойне уверен, что я совершаю или откатываю все транзакции.
Вот мой hibernate.cfg.xml
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-configuration PUBLIC "-//Hibernate/Hibernate Configuration DTD 3.0//EN" "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<session-factory>
<property name="hibernate.bytecode.use_reflection_optimizer">false</property>
<property name="hibernate.connection.driver_class">org.gjt.mm.mysql.Driver</property>
<property name="hibernate.connection.password"></property>
<property name="hibernate.connection.url">jdbc:mysql://localhost/test</property>
<property name="hibernate.connection.username">root</property>
<property name="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</property>
<property name="hibernate.transaction.factory_class">org.hibernate.transaction.JDBCTransactionFactory</property>
<property name="hibernate.current_session_context_class">thread</property>
<!--property name="hibernate.show_sql">true</property>
<property name="hibernate.format_sql">true</property-->
<property name="c3p0.min_size">3</property>
<property name="c3p0.max_size">5</property>
<property name="c3p0.timeout">1800</property>
<property name="c3p0.preferredTestQuery">SELECT 1</property>
<property name="c3p0.testConnectionOnCheckout">true</property>
<property name="c3p0.idle_test_period">100</property> <!-- seconds -->
<property name="cache.provider_class">org.hibernate.cache.NoCacheProvider</property>
<property name="cache.use_query_cache">false</property>
<property name="cache.use_minimal_puts">false</property>
<property name="max_fetch_depth">10</property>
<property name="hibernate.hbm2ddl.auto">update</property>
<!-- classes removed -->
</session-factory>
Параметр, который, как я думал, фиксировал бы, был c3p0.idle_test_period
- по умолчанию он равен 0. Однако у нас по-прежнему есть проблема с Broken Pipe после 8 часов работы. Несмотря на то, что с помощью Google существует множество сообщений, ни один из них не дает удовлетворительного ответа.
Ответы
Ответ 1
Итак, оказалось, что у меня отсутствовала ключевая строка, которая активировала c3p0 (параметры c3p0, которые я настраивал, не имели никакого эффекта, потому что Hibernate использовал его в пуле соединений, который он соответствующим образом предупреждает, не подходит для производства). В hibernate 2.x установите для параметра hibernate.c3p0.max_size
свойство enabled c3p0 connection pooling. Однако в 3.x вы должны указать следующее свойство -
<property name="hibernate.connection.provider_class">org.hibernate.connection.C3P0ConnectionProvider</property>
Кроме того, вот мои окончательные параметры конфигурации -
<property name="hibernate.c3p0.min_size">3</property>
<property name="hibernate.c3p0.max_size">5</property>
<property name="hibernate.c3p0.timeout">1800</property>
<property name="hibernate.c3p0.idle_test_period">100</property> <!-- seconds -->
Очень жаль, что и Hibernate, и c3p0 имеют ужасную документацию в этом отношении.
Ответ 2
Здесь есть две вещи. Вы должны прочитать эту статью для получения более подробной информации, но примите во внимание:
- Вы можете настроить настройку MySQL wait_timeout на желаемое значение более 8 часов.
- Настройки Hibernate должны включать "спящий режим". перед "c3p0", например.
hibernate.c3p0.idle_test_period
вместо c3p0.idle_test_period
Ответ 3
Это решение, когда у вас сломанная труба из-за комбинации tomcat wait_timeout = 28800 секунд (8h) и maxIdleTime = 0 в c3p0:
Я изменил локальный tomcat wait_timeout через файл my.ini на 120 секунд (2 мин). И я добавил следующее:
maxIdleTime = 100
idleConnectionTestPeriod = 0 (то же, что и значение по умолчанию/как если бы оно не существовало)
другие:
acquireIncrement = 2
minPoolSize = 2
MaxPoolSize = 5
maxIdleTimeExcessConnections = 10
У меня не было проблем с этой настройкой.
Мне не нужно было использовать idleConnectionTestPeriod!
Если tomcat wait_timeout составляет 28800 секунд, а maxIdleTime - 25200, это означает, что c3p0 будет закрывать незанятое соединение в 3600 секунд (1 час) раньше, прежде чем tomcat выдает исключение "сломанный канал". Разве это не так?!
Как вы можете видеть, у меня нет проблем с предоставлением только maxIdleTime.
К сожалению, эти:
maxIdleTime
idleConnectionTestPeriod
configuring_connection_testing
testConnectionOnCheckin
не объясняйте слишком много угловых случаев.
И, кстати, вот как открыть файл tomcat my.ini с помощью Notepad ++:
http://drupal.org/node/32715#comment-4907440
Cheers,
Деспот
Ответ 4
У меня несколько проблем -
- C3P0ConnectionProvider не найден
- Я решаю его, используя версию hibernate c3p0
<groupId>org.hibernate</groupId>
<artifactId>hibernate-entitymanager</artifactId>
<version>3.5.6-Final</version>
</dependency>
<!-- c3p0 -->
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-c3p0</artifactId>
<version>3.3.1.GA</version>
</dependency>
- У меня есть проблема wait_timeout на mysql. Сначала я установил /etc/my.cnf wait_timeout = 10
то я изменил значение времени ожидания в режиме ожидания, чтобы уменьшить значение wait_timeout, которое < 10
Это решило мою проблему.
<property name="hibernate.connection.provider_class" value="org.hibernate.connection.C3P0ConnectionProvider" /> <property name="hibernate.c3p0.acquire_increment" value="1" />
<property name="hibernate.c3p0.idle_test_period" value="28690"/>
<property name="hibernate.c3p0.timeout" value="1800" />
<property name="hibernate.c3p0.max_size" value="5" />
<property name="hibernate.c3p0.min_size" value="3" />
<property name="hibernate.c3p0.max_statement" value="50" />
<property name="hibernate.c3p0.preferredTestQuery" value="select 1;"/>
Ответ 5
У меня возникала такая же проблема, и нам потребовалось время, чтобы выяснить решение.
Я использую Hibernate 4.0.1 и mysql 5.1 (без spring), и я столкнулся с этой проблемой. Сначала убедитесь, что вы правильно сконфигурировали банки c3p0, которые необходимы.
Я использовал эти свойства в файле hibernate.cfg.xml
<property name="hibernate.c3p0.validate">true</property>
<property name="hibernate.connection.provider_class">org.hibernate.service.jdbc.connections.internal.C3P0ConnectionProvider</property>
<property name="hibernate.c3p0.min_size">5</property>
<property name="hibernate.c3p0.max_size">20</property>
<property name="hibernate.c3p0.max_statements">50</property>
<property name="hibernate.c3p0.preferredTestQuery">SELECT 1;</property>
<property name="hibernate.c3p0.testConnectionOnCheckout">true</property>
<property name="hibernate.c3p0.idle_test_period">10</property>
<property name="hibernate.c3p0.acquireRetryAttempts">5</property>
<property name="hibernate.c3p0.acquireRetryDelay">200</property>
<property name="hibernate.c3p0.timeout">40</property>
Но это бесполезно, потому что C3p0 по-прежнему использует свойства по умолчанию, а не свойства, которые я задал в файле hibernate.cfg.xml, вы можете проверить его в журналах. Итак, я искал многие веб-сайты для правильного решения, и, наконец, я придумал это. удалите свойства C3p0 в cfg.xml и создайте c3p0-config.xml в корневом пути (вместе с cfg.xml) и установите свойства следующим образом.
<c3p0-config>
<default-config>
<property name="automaticTestTable">con_test</property>
<property name="checkoutTimeout">40</property>
<property name="idleConnectionTestPeriod">10</property>
<property name="initialPoolSize">10</property>
<property name="maxPoolSize">20</property>
<property name="minPoolSize">5</property>
<property name="maxStatements">50</property>
<property name="preferredTestQuery">SELECT 1;</property>
<property name="acquireRetryAttempts">5</property>
<property name="acquireRetryDelay">200</property>
<property name="maxIdleTime">30</property>
</default-config>
</c3p0-config>
но если вы запустите, ORM принимает соединение jdbc, но не пул соединений C3p0, потому что мы должны добавить эти свойства в файл hibernate.cfg.xml
<property name="hibernate.c3p0.validate">true</property>
<property name="hibernate.connection.provider_class">org.hibernate.service.jdbc.connections.internal.C3P0ConnectionProvider</property>
теперь все работает нормально (по крайней мере, это сработало для меня), и проблема решена.
проверьте ссылки для ссылок.
http://www.mchange.com/projects/c3p0/index.html#configuring_connection_testing
https://community.jboss.org/wiki/HowToConfigureTheC3P0ConnectionPool
Надеюсь, это решит вашу проблему.