版权声明:本文为博主原创文章,如果转载请给出原文链接:http://doofuu.com/article/4156132.html
问题:erlang热更新会kill掉还在调用'old' 代码的进程吗?
前几天在跟人讨论erlang模块热更新时对方认为不会kill掉还在调用‘old’代码的进程,原因是这会破坏掉erlang的稳定性,这样热更也就完全没有了意义!今天有时间我来写这篇博客验证下。
大家都知道erlang VM为每个模块最多保存2份代码,当前版本'current'和旧版本'old',当模块第一次被加载时,代码就是'current'版本。如果有新的代码被加载,'current'版本代码就变成了'old'版本,新的代码就成了'current'版本。erlang用两个版本共存的方法来保证任何时候总有一个版本可用,对外服务就不会停止。
测试代码如下:
-module(test). -compile(export_all). start() -> Pid = spawn(fun() -> do_loop() end), register(?MODULE, Pid). do_loop() -> receive Msg -> io:format("~p~n", [Msg]) end, do_loop(). hot_update(Mod) -> code:purge(Mod), code:load_file(Mod).
测试结果如下:
从结果可以看出erlang热更时会kill掉一直使用‘old’代码的进程,原因是如果进程一直在自己loop里面,就会一直跑着‘old‘版本的代码, 那么如何解决这个问题呢? 请接着看下面
代码修改如下:
-module(test). -compile(export_all). start() -> Pid = spawn(fun() -> do_loop() end), register(?MODULE, Pid). do_loop() -> receive code_switch -> ?MODULE:do_loop(); Msg -> io:format("~p~n", [Msg]), do_loop() end. hot_update(Mod) -> code:purge(Mod), code:load_file(Mod).
测试结果如下:
在每次更新完之后给进程发送一条code_switch消息, 这样就没问题了,但是为什么呢? 这就涉及到erlang函数内部调用和外部调用的问题了,详细请参考下面几篇文章:
http://blog.csdn.net/mycwq/article/details/43372687
http://blog.csdn.net/mycwq/article/details/41175237
http://erlang.org/doc/reference_manual/code_loading.html#id86381
http://learnyousomeerlang.com/designing-a-concurrent-application#hot-code-loving
共有 0 条评论 - Erlang热更新会kill掉阻塞进程吗?