tomcat OutOfMemoryError出现的三种情况解决办法

本文分析了Tomcat在生产环境中常见的三种内存溢出问题及其解决方案,包括堆溢出、永久保存区域溢出及无法创建新线程等问题,并提供了具体的JVM参数调整建议。

Tomcat内存溢出的原因

在生产环境中tomcat内存设置不好很容易出现内存溢出。造成内存原因是不一样的,当然处理方式也不一样。

这里根据平时遇到的情况和相关资料进行一个总结。常见的一般会有下面三种情况:

1.OutOfMemoryError: Java heap space

2.OutOfMemoryError: PermGen space

3.OutOfMemoryError: unable to create new native thread.

  Tomcat内存溢出解决方案

对于前两种情况,在应用本身没有内存泄露的情况下可以用设置tomcat jvm参数来解决。(-Xms -Xmx -XX:PermSize  -XX:MaxPermSize)

最后一种可能需要调整操作系统和tomcat jvm参数同时调整才能达到目的。

第一种:是堆溢出。

在JVM中如果98%的时间是用于GC且可用的 Heap size 不足2%的时候将抛出此异常信息。

没有内存泄露的情况下,调整-Xms -Xmx参数可以解决。

-Xms:初始堆大小

-Xmx:最大堆大小

但堆的大小受下面三方面影响:

1.相关操作系统的数据模型(32-bt还是64-bit)限制;(32位系统下,一般限制在1.5G~2G;我在2003 server 系统下(物理内存:4G和6G,jdk:1.6)测试 1612M,64为操作系统对内存无限制。)

2.系统的可用虚拟内存限制;

3.系统的可用物理内存限制。

堆的大小可以使用 java -Xmx***M  version 命令来测试 。支持的话会出现jdk的 版本号,不支持会报错。

-Xms -Xmx一般配置成一样比较好比如set JAVA_OPTS= -Xms1024m -Xmx1024m

第二种:永久保存区域溢出

PermGen space的全称是Permanent Generation space,是指内存的永久保存区域。这一部分用于存放Class和Meta的信息,Class在被 Load的时候被放入PermGen space区域,它和和存放Instance的Heap区域不同,GC(Garbage Collection)不会在主程序运行期对PermGen space进行清理,所以如果你的APP会LOAD很多CLASS的话,就很可能出现PermGen space错误。这种错误常见在web服务器 对JSP进行pre compile的时候。但目前的hibernate和spring项目中也很容易出现这样的问题。http://www.javaeye.com/topic/80620 ?page=1 的帖子有讨论的这个问题。可能是由于这些框架会动态class,而且jvm的gc是不会清理PemGen space的,导致内存溢出。

这一个一般是加大-XX:PermSize  -XX:MaxPermSize 来解决问题。

-XX:PermSize 永久保存区域初始大小

-XX:PermSize 永久保存区域初始最大值

这一般结合第一条使用,比如 set JAVA_OPTS= -Xms1024m -Xmx1024m  -XX:PermSize=128M -XX:PermSize=256M

有一点需要注意:java -Xmx***M  version 命令来测试的最大堆内存是 -Xmx与 -XX:PermSize的 和 比如系统支持最大的jvm堆大小事1.5G,那  -Xmx1024m  -XX:PermSize=768M 是无法运行的。

原因: 
      PermGen space的全称是Permanent Generation space,是指内存的永久保存区域,这块内存主要是被JVM存放Class和Meta信息的,Class在被Loader时就会被放到PermGen space中,它和存放类实例(Instance)的Heap区域不同,GC(Garbage Collection)不会在主程序运行期对PermGen space进行清理,所以如果你的应用中有很CLASS的话,就很可能出现PermGen space错误,这种错误常见在web服务器对JSP进行pre compile的时候。如果你的WEB APP下都用了大量的第三方jar, 其大小超过了jvm默认的大小(4M)那么就会产生此错误信息了。

解决方法:

1. 手动设置MaxPermSize大小
      修改TOMCAT_HOME/bin/catalina.bat(Linux下为catalina.sh),在“echo "Using CATALINA_BASE:   $CATALINA_BASE"”上面加入以下行:
set JAVA_OPTS=%JAVA_OPTS% -server -XX:PermSize=128M -XX:MaxPermSize=512m

catalina.sh下为:
JAVA_OPTS="$JAVA_OPTS -server -XX:PermSize=128M -XX:MaxPermSize=512m"


2. 将相同的第三方jar文件移置到tomcat/shared/lib目录下,这样可以达到减少jar 文档重复占用内存的目的。


如果遇到这个异常:java.lang.OutOfMemoryError: Java heap space 是什么原因呢?

解释: 
Heap size 设置
JVM堆的设置是指java程序运行过程中JVM可以调配使用的内存空间的设置.JVM在启动的时候会自动设置Heap size的值,其初始空间(即-Xms)是物理内存的1/64,最大空间(-Xmx)是物理内存的1/4。可以利用JVM提供的-Xmn -Xms -Xmx等选项可进行设置。Heap size 的大小是Young Generation 和Tenured Generaion 之和。
提示:在JVM中如果98%的时间是用于GC且可用的Heap size 不足2%的时候将抛出此异常信息。
提示:Heap Size 最大不要超过可用物理内存的80%,一般的要将-Xms和-Xmx选项设置为相同,而-Xmn为1/4的-Xmx值。

解决方法: 
      手动设置Heap size
      修改TOMCAT_HOME/bin/catalina.bat,在“echo "Using CATALINA_BASE:   $CATALINA_BASE"”上面加入以下行:
set JAVA_OPTS=%JAVA_OPTS% -server -Xms800m -Xmx800m   -XX:MaxNewSize=256m

或修改catalina.sh
在“echo "Using CATALINA_BASE:   $CATALINA_BASE"”上面加入以下行:
JAVA_OPTS="$JAVA_OPTS  -server -Xms800m -Xmx800m   -XX:MaxNewSize=256m"

另外看到了另外一个帖子,觉得挺好,摘抄如下:

 主题: 分析java.lang.OutOfMemoryError: PermGen space

SUN JDK+Tomcat 5.5.20运行服务的时候遇到问题,服务器跑几天后就会挂掉,并报java.lang.OutOfMemoryError: PermGen space异常。


发现很多人把问题归因于: spring,hibernate,tomcat,因为他们动态产生类,导致JVM中的permanent heap溢出 。然后解决方法众说纷纭,有人说升级 tomcat版本到最新甚至干脆不用tomcat。还有人怀疑spring的问题,在spring论坛上讨论很激烈,因为spring在AOP时使用CBLIB会动态产生很多类。

但问题是为什么这些王牌的开源会出现同一个问题呢,那么是不是更基础的原因呢?tomcat在Q&A很隐晦的回答了这一点,我们知道这个问题,但这个问题是由一个更基础的问题产生。

于是有人对更基础的JVM做了检查,发现了问题的关键。原来SUN 的JVM把内存分了不同的区,其中一个就是permenter区用来存放用得非常多的类和类描述。本来SUN设计的时候认为这个区域在JVM启动的时候就固定了,但他没有想到现在动态会用得这么广泛。而且这个区域有特殊的垃圾收回机制,现在的问题是动态加载类到这个区域后,gc根本没办法回收!

2003年的时候就有一个bug报告给sun,但是到现在,这个bug还没有close!有人在这个bug加了句评语:“A bug this critical is open since 2003? Absolutely shameful.” 我觉得SUN在这个BUG上确实有些丢脸。

对这个bug最彻底的解决办法就是不要用SUN的JDK,而改用BEA的 JRokit.

打不过,还逃不过吗? 有众多的选择,这就是开源的好。 :)


第三种:无法创建新的线程。

这种现象比较少见,也比较奇怪,主要是和jvm与系统内存的比例有关。

这种怪事是因为JVM已经被系统分配了大量的内存(比如1.5G),并且它至少要占用可用内存的一半。有人发现,在线程个数很多的情况下,你分配给JVM 的内存越多,那么,上述错误发生的可能性就越大。

产生这种现象的原因如下(从这个blog中了解到原因:http://hi.baidu.com/hexiong/blog/item/16dc9e518fb10c2542a75b3c.html ):

每一个32位的进程最多可以使用2G的可用内存,因为另外2G被操作系统保留。这里假设使用1.5G给JVM,那么还余下500M可用内存。这500M内 存中的一部分必须用于系统dll的加载,那么真正剩下的也许只有400M,现在关键的地方出现了:当你使用Java 创建一个线程,在JVM的内存里也会创建一个Thread对象,但是同时也会在操作系统里创建一个真正 的物理线程(参考JVM规范),操作系统会在余下的400兆内存里创建这个物理线程,而不是在JVM的1500M的内存堆里创建。在jdk1.4里头,默 认的栈大小是256KB,但是在jdk1.5里头,默认的栈大小为1M每线程,因此,在余下400M的可用内存里边我们最多也只能创建400个可用线程。

这样结论就出来了,要想创建更多的线程,你必须减少分配给JVM的最大内存。还有一种做法是让JVM宿主在你的JNI代码里边。

给出一个有关能够创建线程的最大个数的估算公式:

(MaxProcessMemory - JVMMemory - ReservedOsMemory) / (ThreadStackSize) = Number of threads

对于jdk1.5而言,假设操作系统保留120M内存:

1.5GB JVM: (2GB-1.5Gb-120MB)/(1MB) = ~380 threads

1.0GB JVM: (2GB-1.0Gb-120MB)/(1MB) = ~880 threads

在2000/XP/2003的boot.ini里头有一个启动选项,好像是:/PAE /3G ,可以让用户进程最大内存扩充至3G,这时操作系统只能占用最多1G的虚存。那样应该可以让JVM创建更多的线程。

因此这种情况需要结合操作系统进行相关调整。

因此:我们需要结合不同情况对tomcat内存分配进行不同的诊断才能从根本上解决问题。

at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invok (NativeMethodAccessorImpl.java:62) at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.i voke(DelegatingMethodAccessorImpl.java:43) at java.base/java.lang.reflect.Method.invoke(Method.java:566) at org.apache.catalina.startup.Bootstrap.start(Bootstrap.java:34 ) at org.apache.catalina.startup.Bootstrap.main(Bootstrap.java:473 Caused by: org.apache.catalina.LifecycleException: 鏃犳硶鍚姩缁勪欢[St ndardEngine[Catalina].StandardHost[localhost].StandardContext[/KuCun2]] at org.apache.catalina.util.LifecycleBase.handleSubClassExceptio (LifecycleBase.java:440) at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.ja a:198) at org.apache.catalina.core.ContainerBase.addChildInternal(Conta nerBase.java:717) ... 37 more Caused by: org.springframework.beans.factory.BeanCreationException: Erro creating bean with name 'entityManagerFactory' defined in class path resource org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaConfiguration.class] Invocation of init method failed; nested exception is javax.persistence.Persis enceException: [PersistenceUnit: default] Unable to build Hibernate SessionFact ry; nested exception is org.hibernate.exception.JDBCConnectionException: Unable to open JDBC Connection for DDL execution at org.springframework.beans.factory.support.AbstractAutowireCap bleBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1799) at org.springframework.beans.factory.support.AbstractAutowireCap bleBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:594) at org.springframework.beans.factory.support.AbstractAutowireCap bleBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:516) at org.springframework.beans.factory.support.AbstractBeanFactory lambda$doGetBean$0(AbstractBeanFactory.java:324) at org.springframework.beans.factory.support.DefaultSingletonBea Registry.getSingleton(DefaultSingletonBeanRegistry.java:234) at org.springframework.beans.factory.support.AbstractBeanFactory doGetBean(AbstractBeanFactory.java:322) at org.springframework.beans.factory.support.AbstractBeanFactory getBean(AbstractBeanFactory.java:202) at org.springframework.context.support.AbstractApplicationContex .getBean(AbstractApplicationContext.java:1109) at org.springframework.context.support.AbstractApplicationContex .finishBeanFactoryInitialization(AbstractApplicationContext.java:869) at org.springframework.context.support.AbstractApplicationContex .refresh(AbstractApplicationContext.java:551) at org.springframework.boot.web.servlet.context.ServletWebServer pplicationContext.refresh(ServletWebServerApplicationContext.java:143) at org.springframework.boot.SpringApplication.refresh(SpringAppl cation.java:755) at org.springframework.boot.SpringApplication.refresh(SpringAppl cation.java:747) at org.springframework.boot.SpringApplication.refreshContext(Spr ngApplication.java:402) at org.springframework.boot.SpringApplication.run(SpringApplicat on.java:312) at org.springframework.boot.web.servlet.support.SpringBootServle Initializer.run(SpringBootServletInitializer.java:173) at org.springframework.boot.web.servlet.support.SpringBootServle Initializer.createRootApplicationContext(SpringBootServletInitializer.java:153) at org.springframework.boot.web.servlet.support.SpringBootServle Initializer.onStartup(SpringBootServletInitializer.java:95) at org.springframework.web.SpringServletContainerInitializer.onS artup(SpringServletContainerInitializer.java:172) at org.apache.catalina.core.StandardContext.startInternal(Standa dContext.java:5128) at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.ja a:183) ... 38 more Caused by: javax.persistence.PersistenceException: [PersistenceUnit: def ult] Unable to build Hibernate SessionFactory; nested exception is org.hibernat .exception.JDBCConnectionException: Unable to open JDBC Connection for DDL exec tion at org.springframework.orm.jpa.AbstractEntityManagerFactoryBean. uildNativeEntityManagerFactory(AbstractEntityManagerFactoryBean.java:403) at org.springframework.orm.jpa.AbstractEntityManagerFactoryBean. fterPropertiesSet(AbstractEntityManagerFactoryBean.java:378) at org.springframework.orm.jpa.LocalContainerEntityManagerFactor Bean.afterPropertiesSet(LocalContainerEntityManagerFactoryBean.java:341) at org.springframework.beans.factory.support.AbstractAutowireCap bleBeanFactory.invokeInitMethods(AbstractAutowireCapableBeanFactory.java:1858) at org.springframework.beans.factory.support.AbstractAutowireCap bleBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1795) ... 58 more Caused by: org.hibernate.exception.JDBCConnectionException: Unable to op n JDBC Connection for DDL execution at org.hibernate.exception.internal.SQLStateConversionDelegate.c nvert(SQLStateConversionDelegate.java:112) at org.hibernate.exception.internal.StandardSQLExceptionConverte .convert(StandardSQLExceptionConverter.java:42) at org.hibernate.engine.jdbc.spi.SqlExceptionHelper.convert(SqlE ceptionHelper.java:113) at org.hibernate.engine.jdbc.spi.SqlExceptionHelper.convert(SqlE ceptionHelper.java:99) at org.hibernate.resource.transaction.backend.jdbc.internal.DdlT ansactionIsolatorNonJtaImpl.getIsolatedConnection(DdlTransactionIsolatorNonJtaI pl.java:69) at org.hibernate.tool.schema.internal.exec.ImprovedExtractionCon extImpl.getJdbcConnection(ImprovedExtractionContextImpl.java:60) at org.hibernate.tool.schema.internal.exec.ImprovedExtractionCon extImpl.getJdbcDatabaseMetaData(ImprovedExtractionContextImpl.java:67) at org.hibernate.tool.schema.extract.internal.InformationExtract rJdbcDatabaseMetaDataImpl.getTables(InformationExtractorJdbcDatabaseMetaDataImp .java:333) at org.hibernate.tool.schema.extract.internal.DatabaseInformatio Impl.getTablesInformation(DatabaseInformationImpl.java:120) at org.hibernate.tool.schema.internal.GroupedSchemaMigratorImpl. erformTablesMigration(GroupedSchemaMigratorImpl.java:65) at org.hibernate.tool.schema.internal.AbstractSchemaMigrator.per ormMigration(AbstractSchemaMigrator.java:207) at org.hibernate.tool.schema.internal.AbstractSchemaMigrator.doM gration(AbstractSchemaMigrator.java:114) at org.hibernate.tool.schema.spi.SchemaManagementToolCoordinator performDatabaseAction(SchemaManagementToolCoordinator.java:184) at org.hibernate.tool.schema.spi.SchemaManagementToolCoordinator process(SchemaManagementToolCoordinator.java:73) at org.hibernate.internal.SessionFactoryImpl.<init>(SessionFacto yImpl.java:318) at org.hibernate.boot.internal.SessionFactoryBuilderImpl.build(S ssionFactoryBuilderImpl.java:468) at org.hibernate.jpa.boot.internal.EntityManagerFactoryBuilderIm l.build(EntityManagerFactoryBuilderImpl.java:1259) at org.springframework.orm.jpa.vendor.SpringHibernateJpaPersiste ceProvider.createContainerEntityManagerFactory(SpringHibernateJpaPersistencePro ider.java:58) at org.springframework.orm.jpa.LocalContainerEntityManagerFactor Bean.createNativeEntityManagerFactory(LocalContainerEntityManagerFactoryBean.ja a:365) at org.springframework.orm.jpa.AbstractEntityManagerFactoryBean. uildNativeEntityManagerFactory(AbstractEntityManagerFactoryBean.java:391) ... 62 more Caused by: com.mysql.cj.jdbc.exceptions.CommunicationsException: Communi ations link failure he last packet sent successfully to the server was 0 milliseconds ago. The driv r has not received any packets from the server. at com.mysql.cj.jdbc.exceptions.SQLError.createCommunicationsExc ption(SQLError.java:175) at com.mysql.cj.jdbc.exceptions.SQLExceptionsMapping.translateEx eption(SQLExceptionsMapping.java:64) at com.mysql.cj.jdbc.ConnectionImpl.createNewIO(ConnectionImpl.j va:825) at com.mysql.cj.jdbc.ConnectionImpl.<init>(ConnectionImpl.java:4 6) at com.mysql.cj.jdbc.ConnectionImpl.getInstance(ConnectionImpl.j va:239) at com.mysql.cj.jdbc.NonRegisteringDriver.connect(NonRegistering river.java:188) at com.zaxxer.hikari.util.DriverDataSource.getConnection(DriverD taSource.java:138) at com.zaxxer.hikari.pool.PoolBase.newConnection(PoolBase.java:3 8) at com.zaxxer.hikari.pool.PoolBase.newPoolEntry(PoolBase.java:20 ) at com.zaxxer.hikari.pool.HikariPool.createPoolEntry(HikariPool. ava:477) at com.zaxxer.hikari.pool.HikariPool.checkFailFast(HikariPool.ja a:560) at com.zaxxer.hikari.pool.HikariPool.<init>(HikariPool.java:115) at com.zaxxer.hikari.HikariDataSource.getConnection(HikariDataSo rce.java:112) at org.hibernate.engine.jdbc.connections.internal.DatasourceConn ctionProviderImpl.getConnection(DatasourceConnectionProviderImpl.java:122) at org.hibernate.engine.jdbc.env.internal.JdbcEnvironmentInitiat r$ConnectionProviderJdbcConnectionAccess.obtainConnection(JdbcEnvironmentInitia or.java:180) at org.hibernate.resource.transaction.backend.jdbc.internal.DdlT ansactionIsolatorNonJtaImpl.getIsolatedConnection(DdlTransactionIsolatorNonJtaI pl.java:43) ... 77 more Caused by: com.mysql.cj.exceptions.CJCommunicationsException: Communicat ons link failure he last packet sent successfully to the server was 0 milliseconds ago. The driv r has not received any packets from the server. at java.base/jdk.internal.reflect.NativeConstructorAccessorImpl. ewInstance0(Native Method) at java.base/jdk.internal.reflect.NativeConstructorAccessorImpl. ewInstance(NativeConstructorAccessorImpl.java:62) at java.base/jdk.internal.reflect.DelegatingConstructorAccessorI pl.newInstance(DelegatingConstructorAccessorImpl.java:45) at java.base/java.lang.reflect.Constructor.newInstance(Construct r.java:490) at com.mysql.cj.exceptions.ExceptionFactory.createException(Exce tionFactory.java:62) at com.mysql.cj.exceptions.ExceptionFactory.createException(Exce tionFactory.java:105) at com.mysql.cj.exceptions.ExceptionFactory.createException(Exce tionFactory.java:150) at com.mysql.cj.exceptions.ExceptionFactory.createCommunications xception(ExceptionFactory.java:166) at com.mysql.cj.protocol.a.NativeProtocol.negotiateSSLConnection NativeProtocol.java:379) at com.mysql.cj.protocol.a.NativeAuthenticationProvider.connect( ativeAuthenticationProvider.java:206) at com.mysql.cj.protocol.a.NativeProtocol.connect(NativeProtocol java:1430) at com.mysql.cj.NativeSession.connect(NativeSession.java:134) at com.mysql.cj.jdbc.ConnectionImpl.connectOneTryOnly(Connection mpl.java:945) at com.mysql.cj.jdbc.ConnectionImpl.createNewIO(ConnectionImpl.j va:815) ... 90 more Caused by: javax.net.ssl.SSLHandshakeException: NotBefore: Fri Jun 27 18 58:12 CST 2025 at java.base/sun.security.ssl.Alert.createSSLException(Alert.jav :128) at java.base/sun.security.ssl.TransportContext.fatal(TransportCo text.java:321) at java.base/sun.security.ssl.TransportContext.fatal(TransportCo text.java:264) at java.base/sun.security.ssl.TransportContext.fatal(TransportCo text.java:259) at java.base/sun.security.ssl.CertificateMessage$T13CertificateC nsumer.checkServerCerts(CertificateMessage.java:1329) at java.base/sun.security.ssl.CertificateMessage$T13CertificateC nsumer.onConsumeCertificate(CertificateMessage.java:1204) at java.base/sun.security.ssl.CertificateMessage$T13CertificateC nsumer.consume(CertificateMessage.java:1151) at java.base/sun.security.ssl.SSLHandshake.consume(SSLHandshake. ava:392) at java.base/sun.security.ssl.HandshakeContext.dispatch(Handshak Context.java:444) at java.base/sun.security.ssl.HandshakeContext.dispatch(Handshak Context.java:421) at java.base/sun.security.ssl.TransportContext.dispatch(Transpor Context.java:178) at java.base/sun.security.ssl.SSLTransport.decode(SSLTransport.j va:164) at java.base/sun.security.ssl.SSLSocketImpl.decode(SSLSocketImpl java:1152) at java.base/sun.security.ssl.SSLSocketImpl.readHandshakeRecord( SLSocketImpl.java:1063) at java.base/sun.security.ssl.SSLSocketImpl.startHandshake(SSLSo ketImpl.java:402) at com.mysql.cj.protocol.ExportControlled.performTlsHandshake(Ex ortControlled.java:347) at com.mysql.cj.protocol.StandardSocketFactory.performTlsHandsha e(StandardSocketFactory.java:191) at com.mysql.cj.protocol.a.NativeSocketConnection.performTlsHand hake(NativeSocketConnection.java:101) at com.mysql.cj.protocol.a.NativeProtocol.negotiateSSLConnection NativeProtocol.java:370) ... 95 more Caused by: java.security.cert.CertificateNotYetValidException: NotBefore Fri Jun 27 18:58:12 CST 2025 at java.base/sun.security.x509.CertificateValidity.valid(Certifi ateValidity.java:270) at java.base/sun.security.x509.X509CertImpl.checkValidity(X509Ce tImpl.java:687) at java.base/sun.security.x509.X509CertImpl.checkValidity(X509Ce tImpl.java:660) at com.mysql.cj.protocol.ExportControlled$X509TrustManagerWrappe .checkServerTrusted(ExportControlled.java:392) at java.base/sun.security.ssl.AbstractTrustManagerWrapper.checkS rverTrusted(SSLContextImpl.java:1509) at java.base/sun.security.ssl.CertificateMessage$T13CertificateC nsumer.checkServerCerts(CertificateMessage.java:1313) ... 109 more 3-Jun-2012 02:43:37.673 淇℃伅 [main] org.apache.catalina.startup.HostConfig.de loyWAR web搴旂敤绋嬪簭瀛樻。鏂囦欢[D:\apache-tomcat-9.0.37\webapps\KuCun2.war] 鐨勯儴缃插凡鍦╗34,866]ms鍐呭畬鎴? 3-Jun-2012 02:43:37.673 淇℃伅 [main] org.apache.catalina.startup.HostConfig.de loyDirectory 鎶妛eb 搴旂敤绋嬪簭閮ㄧ讲鍒扮洰褰?[D:\apache-tomcat-9.0.37\webapps docs] 3-Jun-2012 02:43:37.829 淇℃伅 [main] org.apache.catalina.startup.HostConfig.de loyDirectory Web搴旂敤绋嬪簭鐩綍[D:\apache-tomcat-9.0.37\webapps\docs]鐨勯儴 缃插凡鍦╗156]姣鍐呭畬鎴? 3-Jun-2012 02:43:37.829 淇℃伅 [main] org.apache.catalina.startup.HostConfig.de loyDirectory 鎶妛eb 搴旂敤绋嬪簭閮ㄧ讲鍒扮洰褰?[D:\apache-tomcat-9.0.37\webapps examples] 3-Jun-2012 02:43:39.826 淇℃伅 [main] org.apache.catalina.startup.HostConfig.de loyDirectory Web搴旂敤绋嬪簭鐩綍[D:\apache-tomcat-9.0.37\webapps\examples]鐨 勯儴缃插凡鍦╗1,997]姣鍐呭畬鎴? 3-Jun-2012 02:43:39.842 淇℃伅 [main] org.apache.catalina.startup.HostConfig.de loyDirectory 鎶妛eb 搴旂敤绋嬪簭閮ㄧ讲鍒扮洰褰?[D:\apache-tomcat-9.0.37\webapps host-manager] 3-Jun-2012 02:43:39.982 淇℃伅 [main] org.apache.catalina.startup.HostConfig.de loyDirectory Web搴旂敤绋嬪簭鐩綍[D:\apache-tomcat-9.0.37\webapps\host-manager 鐨勯儴缃插凡鍦╗140]姣鍐呭畬鎴? 3-Jun-2012 02:43:39.982 淇℃伅 [main] org.apache.catalina.startup.HostConfig.de loyDirectory 鎶妛eb 搴旂敤绋嬪簭閮ㄧ讲鍒扮洰褰?[D:\apache-tomcat-9.0.37\webapps manager] 3-Jun-2012 02:43:40.091 淇℃伅 [main] org.apache.catalina.startup.HostConfig.de loyDirectory Web搴旂敤绋嬪簭鐩綍[D:\apache-tomcat-9.0.37\webapps\manager]鐨勯 儴缃插凡鍦╗109]姣鍐呭畬鎴? 3-Jun-2012 02:43:40.091 淇℃伅 [main] org.apache.catalina.startup.HostConfig.de loyDirectory 鎶妛eb 搴旂敤绋嬪簭閮ㄧ讲鍒扮洰褰?[D:\apache-tomcat-9.0.37\webapps ROOT] 3-Jun-2012 02:43:40.138 淇℃伅 [main] org.apache.catalina.startup.HostConfig.de loyDirectory Web搴旂敤绋嬪簭鐩綍[D:\apache-tomcat-9.0.37\webapps\ROOT]鐨勯儴 缃插凡鍦╗47]姣鍐呭畬鎴? 3-Jun-2012 02:43:40.154 淇℃伅 [main] org.apache.coyote.AbstractProtocol.start 寮€濮嬪崗璁鐞嗗彞鏌刐"http-nio-8080"] 3-Jun-2012 02:43:40.200 淇℃伅 [main] org.apache.coyote.AbstractProtocol.start 寮€濮嬪崗璁鐞嗗彞鏌刐"https-jsse-nio-8443"] 3-Jun-2012 02:43:40.216 淇℃伅 [main] org.apache.catalina.startup.Catalina.star [37875]姣鍚庢湇鍔″櫒鍚姩
07-18
<think>我们面对的是一个Tomcat启动时的错误日志。从日志中可以看出,主要问题是部署一个名为`KuCun2`的Web应用时失败,具体原因是无法建立数据库连接,进而导致Hibernate SessionFactory创建失败,最终导致应用启动失败。 错误链如下: 1. 最根本的异常是`javax.net.ssl.SSLHandshakeException`,原因是一个证书的生效日期是2025年(NotBefore: Fri Jun 27 18:58:12 CST 2025),而当前时间(日志显示为2012年)还没有到这个日期,所以证书尚未生效。 2. 这个SSL握手异常导致了`com.mysql.cj.exceptions.CJCommunicationsException`,进而导致`com.mysql.cj.jdbc.exceptions.CommunicationsException`,最终表现为JDBC连接无法打开。 此外,日志中还显示: ``` Caused by: java.security.cert.CertificateNotYetValidException: NotBefore: Fri Jun 27 18:58:12 CST 2025 ``` 所以,问题的核心是:MySQL连接尝试使用SSL,但是提供的证书尚未生效(因为系统时间在证书生效之前)。 解决方案: 1. 检查系统时间:日志中的时间戳显示为“3-Jun-2012”,这明显是一个过去的时间(当前是2023年或之后)。因此,首先需要修正系统时间,确保服务器的时间是准确的。 2. 如果修正系统时间后问题仍然存在,或者暂时无法修正系统时间,可以考虑暂时禁用SSL连接(如果数据库连接不要求SSL)。 禁用SSL的方法: 在JDBC连接字符串中添加参数:`useSSL=false`。例如: ``` jdbc:mysql://localhost:3306/dbname?useSSL=false ``` 但是注意,根据MySQL版本不同,可能还需要考虑其他参数(如`allowPublicKeyRetrieval`等),具体根据情况而定。 另外,从日志中可以看出,这个应用是使用Spring Boot构建的,数据库连接池使用的是Hikari。因此,我们需要检查应用的配置文件(如application.properties或application.yml)中关于数据库连接的配置。 例如,在application.properties中修改: ```properties spring.datasource.url=jdbc:mysql://localhost:3306/your_database?useSSL=false ``` 如果问题是由于证书信任问题,也可以考虑设置`trustCertificateKeyStoreUrl`和`trustCertificateKeyStorePassword`,但在这个案例中,根本原因是证书尚未生效,所以禁用SSL或修正系统时间是更直接的方法。 步骤总结: 1. 修正服务器系统时间,设置为当前正确时间。 2. 如果修正时间后仍然有问题,或者无法修正时间(比如测试环境),则在数据库连接字符串中禁用SSL。 如果应用已经部署到Tomcat,那么修改应用配置文件后需要重新打包并部署。 但是,请注意:禁用SSL会降低连接的安全性,因此只建议在测试或内部网络中使用。生产环境中应确保证书的有效性,并保持系统时间正确。 下面我们给出一个修改Spring Boot应用配置文件的例子: 假设原配置文件(application.properties)中数据库连接配置如下: ```properties spring.datasource.url=jdbc:mysql://localhost:3306/ku_cun spring.datasource.username=root spring.datasource.password=123456 spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver ``` 修改后: ```properties spring.datasource.url=jdbc:mysql://localhost:3306/ku_cun?useSSL=false # 其他不变 ``` 或者,如果使用YAML格式(application.yml): ```yaml spring: datasource: url: jdbc:mysql://localhost:3306/ku_cun?useSSL=false username: root password: 123456 driver-class-name: com.mysql.cj.jdbc.Driver ``` 修改后重新打包应用,并重新部署到Tomcat。 另外,请注意Tomcat的日志中出现了中文字符乱码,这可能是由于系统编码或Tomcat日志编码设置问题,但这不是导致启动失败的原因,可以暂时忽略。 如果问题仍然存在,请检查: - 数据库服务是否正常运行 - 数据库连接URL、用户名、密码是否正确 - 网络是否可达 但是根据错误信息,我们已经定位到是SSL证书时间问题,所以按照上述步骤应该可以解决。 由于我们无法直接修改代码,这里不提供代码修改,而是提供配置文件的修改示例。 但是,如果问题是由于系统时间错误引起的,那么修改系统时间是最根本的解决方案。在Linux服务器上,可以使用`ntpdate`命令同步时间;在Windows服务器上,可以在设置中调整时间。 下面给出一个在Linux上同步时间的命令示例(需要root权限): ```bash ntpdate ntp.ubuntu.com ``` 或者使用`timedatectl`(适用于systemd的系统): ```bash timedatectl set-ntp true timedatectl set-timezone Asia/Shanghai # 设置时区 timedatectl set-time "2023-05-30 10:00:00" # 如果需要手动设置时间,但通常我们使用NTP同步 ``` 在Windows上,可以通过设置->日期和时间->自动设置时间,来同步时间。 综上所述,请先尝试修正系统时间,如果不行再尝试修改数据库连接字符串。 由于我们无法直接操作服务器,所以只能提供这些建议。 最后,请注意日志中还有其他应用部署成功的记录,说明Tomcat本身启动正常,问题只出在`KuCun2`这个应用上。 如果问题解决,应该可以看到应用成功启动的日志。 下面,我们提出一些相关问题:</think>根据提供的错误日志,这是一个典型的Spring Boot应用部署到Tomcat时出现的数据库连接问题。核心问题是**SSL证书未生效导致JDBC连接失败**。以下是详细分析和解决方案: ### 问题分析 1. **根本原因**:MySQL连接使用了SSL证书,但证书生效日期为`Fri Jun 27 18:58:12 CST 2025`,而服务器当前时间为`2012年`(日志中时间戳为`3-Jun-2012`),导致证书验证失败。 ```java Caused by: java.security.cert.CertificateNotYetValidException: NotBefore: Fri Jun 27 18:58:12 CST 2025 ``` 2. **错误链**: - 证书未生效 → SSL握手失败 → MySQL连接失败 → Hibernate初始化失败 → Spring启动失败 3. **关键错误**: ```java javax.persistence.PersistenceException: [PersistenceUnit: default] Unable to build Hibernate SessionFactory; nested exception is org.hibernate.exception.JDBCConnectionException: Unable to open JDBC Connection for DDL execution ``` ### 解决方案 #### 方法1:修正服务器时间(推荐) 服务器时间(2012年)与证书要求时间(2025年)不匹配,需同步正确时间: ```bash # Linux系统时间同步 sudo timedatectl set-ntp true sudo timedatectl set-timezone Asia/Shanghai sudo systemctl restart systemd-timesyncd # Windows系统 # 1. 控制面板 -> 日期和时间 -> 自动设置时间 # 2. 以管理员身份运行命令提示符: w32tm /resync ``` #### 方法2:禁用MySQL SSL连接 在JDBC连接字符串中添加`useSSL=false`参数: ```properties # application.properties 修改示例 spring.datasource.url=jdbc:mysql://localhost:3306/your_db?useSSL=false&serverTimezone=UTC ``` #### 方法3:信任无效证书(不推荐) 在连接字符串中跳过证书验证(仅限测试环境): ```properties spring.datasource.url=jdbc:mysql://localhost:3306/your_db?useSSL=true&allowPublicKeyRetrieval=true&disableSslHostnameVerification=true ``` ### 完整配置示例 ```properties # application.properties spring.datasource.url=jdbc:mysql://localhost:3306/ku_cun?useSSL=false&serverTimezone=UTC spring.datasource.username=your_username spring.datasource.password=your_password spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver # 关闭Hibernate自动DDL(可选) spring.jpa.hibernate.ddl-auto=none ``` ### 验证步骤 1. 重启Tomcat服务 2. 检查日志中是否出现: ```log org.apache.catalina.startup.HostConfig.deployDirectory 部署成功 ``` 3. 访问应用健康检查端点:`http://localhost:8080/actuator/health` --- ### 错误日志关键点解释 | 错误摘要 | 说明 | |----------|------| | `CertificateNotYetValidException` | 服务器时间早于证书生效时间 | | `Unable to open JDBC Connection` | 数据库连接失败 | | `Communications link failure` | 网络层连接异常 | | `LifecycleException` | Tomcat容器启动组件失败 |
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值