erlang热更新会kill掉进程吗?

来源:互联网 发布:windows装mac虚拟机 编辑:程序博客网 时间:2024/04/30 08:51

问题: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


祝生活愉快!

1 0