Dubbo服务的运行方式
1、使用Servlet容器运行(Tomcat、jetty等)
浪费资源增加运维压力(端口管理、容器管理等)
2、自己建立main方法运行Spring容器
这样没有运用到Dubbo本身提供的高级特性,自己写的main方法可能会有缺陷,比如忘记优雅关机
public class App {
private static Logger logger = LoggerFactory.getLogger(App.class);
@Autowired
RedisTemplate redisTemplate;
public static void main(String[] args) {
logger.info("== init context");
try {
final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext(new String[]{"spring/spring-applicationContext.xml"});
context.registerShutdownHook();//可实现优雅关机
context.start();
// 杀java程序执行kill -15 时 会回调此函数 xmlWeb.registerShutdownHook();即实现的此原理去停止spring程序
// Runtime.getRuntime().addShutdownHook(new Thread() {
// @Override
// public void run() {
// logger.info("关程序前处理");
// //TODO... ...关闭mq消费者等... ...
// //正常关闭spring程序
// context.close();
// logger.info("处理完成");
// }
// });
logger.info("== context start");
} catch (Exception e) {
logger.error("== application start error:", e);
return;
}
synchronized (App.class) {
while (true) {
try {
App.class.wait();
} catch (InterruptedException e) {
logger.error("== synchronized error:", e);
}
}
}
}
}
3、使用Dubbo框架提供的main方法运行Spring容器
com.alibaba.dubbo.container.Main 可以实现优雅关机(ShutDownHook)
优雅关机原理:
Dubbo是通过JDK的ShutdownHook来完成优雅停机的
所以如果用户使用 kill -9 PID 等强制关闭命令,是不会执行优雅停机的
只有通过 kill PID时,才会执行
原理:
· 服务提供方
· 停止时,先标记为不接收新请求,新请求过来时直接报错,让客户端重试其他机器。
· 然后,检测线程池中的线程是否正在运行,如果有,等待所有线程执行完成,除非超时,则强制关闭。
· 服务消费方
· 停止时,不在发起新的调用请求,所有新的调用在客户端即报错。
· 然后,检测有没有请求的响应还没有返回,等待响应返回,除非超时,则强制关闭。
设置优雅停机超时时间,缺省超时时间是10秒:(超时则强制关闭)
设置优雅停机超时时间,缺省超时时间是10秒:(超时则强制关闭)
<dubbo:application ...>
<dubbo:parameter key="shutown.timeout" value="60000" /> <!--单位:毫秒-->
</dubbo:application>
如果ShutdownHook不能生效,可以自行调用:ProtocolConfig.destroyAll();