博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
记一次spring cloud踩坑
阅读量:6251 次
发布时间:2019-06-22

本文共 1104 字,大约阅读时间需要 3 分钟。

前言

最近自己尝试着搭建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

转载地址:http://cgysa.baihongyu.com/

你可能感兴趣的文章
多个Img标签之间的间隙处理方法
查看>>
g++ error: expected ‘)’ before ‘*’ token
查看>>
C++的ABI真特么是evil
查看>>
函数声明和函数表达式
查看>>
Matlab基本函数-conj函数
查看>>
linux常用命令 3
查看>>
SharePoint 2013 托管导航 无法被开启的解决办法
查看>>
初识Java Servlet
查看>>
Test1
查看>>
JS图片切换代码合集
查看>>
Aundit使用记录文档
查看>>
原型 、原型链和对象是怎么实现继承的
查看>>
layui中select切换数据_layui 下拉框 动态获取数据
查看>>
佳能hdr_内置HDR功能 佳能5D3特色拍摄功能解析
查看>>
matlab和python转换_将MATLAB代码转换为Python:Python类型和操作顺序
查看>>
jmeter3000用户压测_jmeter集群压测搭建
查看>>
转子接地保护原理_发变组保护动作逻辑
查看>>
hive中groupby优化_面试必备技能-HiveSQL优化
查看>>
uni 页面加载完毕_HTML页面生命周期
查看>>
c语言机票座位预定系统_趁东京奥运!日航要免费送5万张国内机票!给非日本居民...
查看>>