erlang热更新会kill掉进程吗?

    xiaoxiao2021-12-14  22

    我的个人博客网站云诺说上线啦!所有文章都搬到新地址了,点击围观吧!

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

     

    祝生活愉快!

    转载请注明原文地址: https://ju.6miu.com/read-962466.html

    最新回复(0)