记一次spring cloud踩坑

本文深入探讨SpringCloud项目中Eureka客户端在注册中心注册后立即注销的问题,解析根本原因在于JVM随主线程结束而退出。通过添加web依赖或自定义用户线程,确保服务稳定注册。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言

最近自己尝试着搭建springcloud项目,果不其然,刚开始就踩坑了,还是那种一脸懵逼的坑。搭建后程序能正常运行注册到eureka注册中心,但注册好之后便会立即注销掉。刚开始认为是线程抛异常挂掉了,便导致服务注销了。然后各种debug排查,最后发现是正常退出。至于具体原因,我下面做个详细解释。


eureka注册中心


eureka服务端


初次运行结果

这里注册中心启动是完全没问题的,问题就出在client启动,启动日志如下:

看上面的日志可以观察到,服务最开始是有正常注册到eureka注册中心的,但是紧接着会发现。Unregistering application EUREKA-CLIENT with eureka with status DOWN,后面接着 Shutting down DiscoveryClient ... 最后 Unregistering ...


解决方案之一

看日志并未发现很明显的异常,然后debug调试一波,无任何错误发现。便随手google了一下。果然还是有小伙伴曾经也有遇到过同样的问题的。网上的解决办法如下: 在client端pom中加上依赖:

当然,这也只是其中的一个解决办法。其实出现上述情况的原因是JVM随着主线程的结束而退出了。容器close时会触发DiscoveryClient的shutdown方法,便会注销已注册的节点。但是加上web依赖后,就拿tomcat来举例说明,随着springboot项目的启动,会创建一个WebServer:

咱们重点关注下initialize方法中的:

看源码上的注释,便会发现这里开启了一个用户线程(非守护线程),这样主线程退出后JVM也不会退出。也就不会触发注销节点的动作。


其他解决方案

其实解决这个问题的办法还是很简单的,只要你能保证由一个正常的用户线程存在就行了,有可能咱们自己的提供服务的应用不需要用到tomcat这样的web容器,而是像其他rpc服务一样调用时,咱们便可以自己创建一个用户线程即可(默认构造出来的线程都是用户线程,除非你调用setDaemon方法将其daemon属性设置成true才会变成守护线程)。然后咱们自己内部提供一个状态值可以让其正常退出即可。 下面写个简单的例子:


总结

通过这次的踩坑,还是弥补了自己对JVM理解的一些盲区。比如当进程中只要还有一个用户线程存在时,进程便不会立即结束。只有当进程中只存在守护线程或者不存在任何线程的情况下,进程才会马上结束。我们通常了解到的类似gc的操作线程,便是守护线程,这样咱们自己的业务线程在异常情况下挂掉,进程也会随之结束。


End
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值