最近本身尝试着搭建springcloud项目,果不其然,刚开始就踩坑了,仍是那种一脸懵逼的坑。搭建后程序能正常运行注册到eureka注册中心,但注册好以后便会当即注销掉。刚开始认为是线程抛异常挂掉了,便致使服务注销了。而后各类debug排查,最后发现是正常退出。至于具体缘由,我下面作个详细解释。web
这里注册中心启动是彻底没问题的,问题就出在client启动,启动日志以下:spring
看上面的日志能够观察到,服务最开始是有正常注册到eureka注册中心的,可是紧接着会发现。Unregistering application EUREKA-CLIENT with eureka with status DOWN,后面接着 Shutting down DiscoveryClient ... 最后 Unregistering ...。tomcat
看日志并未发现很明显的异常,而后debug调试一波,无任何错误发现。便随手google了一下。果真仍是有小伙伴曾经也有遇到过一样的问题的。网上的解决办法以下:
在client端pom中加上依赖:
固然,这也只是其中的一个解决办法。其实出现上述状况的缘由是JVM随着主线程的结束而退出了。容器close时会触发DiscoveryClient的shutdown方法,便会注销已注册的节点。可是加上web依赖后,就拿tomcat来举例说明,随着springboot项目的启动,会建立一个WebServer:
springboot
我们重点关注下initialize方法中的:
看源码上的注释,便会发现这里开启了一个用户线程(非守护线程),这样主线程退出后JVM也不会退出。也就不会触发注销节点的动做。app
其实解决这个问题的办法仍是很简单的,只要你能保证由一个正常的用户线程存在就好了,有可能我们本身的提供服务的应用不须要用到tomcat这样的web容器,而是像其余rpc服务同样调用时,我们即可以本身建立一个用户线程便可(默认构造出来的线程都是用户线程,除非你调用setDaemon方法将其daemon属性设置成true才会变成守护线程)。而后我们本身内部提供一个状态值可让其正常退出便可。
下面写个简单的例子:
google
经过此次的踩坑,仍是弥补了本身对JVM理解的一些盲区。好比当进程中只要还有一个用户线程存在时,进程便不会当即结束。只有当进程中只存在守护线程或者不存在任何线程的状况下,进程才会立刻结束。咱们一般了解到的相似gc的操做线程,即是守护线程,这样我们本身的业务线程在异常状况下挂掉,进程也会随之结束。线程