20

我正在使用 erlang 进行我的第一个真实项目,但是,为简洁起见,此代码已简化。我希望能够在项目运行时将较新版本的文件远程加载到我的项目中。我读过关于使用类似gen_servergen_fsm免费的行为。虽然这可能会达到结果,但我想用它来学习如何去做,而不仅仅是完成它。我已经阅读了有关代码替换的文档,以及 LYSE 关于Hot Code Loving的一些内容,但我还没有找到任何适合我正在做的事情的东西,所以这是基本的想法。

-module(reloading).

-export([loop/0]).

loop() ->
    receive
        upgrade ->
            ?MODULE:loop();
        hello ->
            io:format("This is a test~n"),
            loop();
        _ ->
            loop()
    end.

我只是在循环我可以发送消息upgrade并且它将加载更新版本的代码的想法。

$ erl
Erlang R15B01 (erts-5.9.1) [source] [64-bit] [async-threads:0] [hipe] [kernel-poll:false]

Eshell V5.9.1  (abort with ^G)
1> c(reloading).
{ok,reloading}
2> Loop = spawn(reloading, loop, []).
<0.39.0>
3> Loop ! hello.
This is a test
hello

此时我将 10 行更改为io:format("I have changed this!~n"),

4> Loop ! upgrade.
upgrade
5> Loop ! hello.  
This is a test
hello

我希望这个电话不会hello打印。我知道我可以简单地调用并按照预期的方式进行这项工作,但我希望向实际项目发送消息,而不是手动更新代码。那么我的断开连接在哪里?我做错了什么,为了热加载这段代码我应该做什么?如前所述,为了教育,我正在寻找非 OTP 解决方案。I have changed this!This is a testc(reloading).

4

4 回答 4

24

为了得到明确的答案,我发布了这个。

使用使用代码@rvirding模块的建议,我将其修改为如下所示:

-module(reloading).

-export([loop/0]).

loop() ->
    receive
        upgrade ->
            code:purge(?MODULE),
            compile:file(?MODULE),
            code:load_file(?MODULE),
            ?MODULE:loop();
        hello ->
            io:format("This is a test~n"),
            loop();
        _ ->
            loop()
    end.

首先code:purge old ?MODULE,然后compile:file新文件,最后code:load_file新文件?MODULE。这按我最初的意图工作。

$ erl
Erlang R15B01 (erts-5.9.1) [source] [64-bit] [async-threads:0] [hipe] [kernel-poll:false]

Eshell V5.9.1  (abort with ^G)
1> Loop = spawn(reloading, loop, []).
<0.34.0>
2> Loop ! hello.
This is a test
hello

将行更改为io:format("I have changed this!~n"),

3> Loop ! upgrade.                   
upgrade
4> Loop ! hello.  
I have changed this!
hello
于 2012-08-15T15:10:59.487 回答
19

虽然 erlang 可以处理模块的两个版本并且调用函数mod:func(...)总是会调用模块的最新版本(如果函数已导出),但您仍然必须将模块的新版本加载到 Erlang 系统中。你不能指望它会自动检测到你碰巧在某个地方有一个新版本的模块,找到它,编译它并加载它。

NB编译和加载是两个独立的事情。因此c(mod). ,两者都编译加载模块,而l(mod).只是加载已编译模块的目标代码(.beam 文件)。Erlang 编译器是从模块中调用的compile,它只是编译并生成一个 .beam 文件,而代码加载由模块处理code

于 2012-08-15T12:42:00.783 回答
1

除了上述之外,我还想注意到一些自动为您重新加载代码的工具。

您应该查看同步活动项目。

于 2015-09-24T19:35:45.017 回答
1

在本地编译 *.beam,然后将其发送到您的服务器并按照手册页中的说明重新加载:

http://erlang.org/documentation/doc-1/reference_manual/code_loading.html#id86381

-module(m).
-export([loop/0]).

loop() ->
    receive
        code_switch ->
            m:loop();
        Msg ->
            ...
            loop()
    end.
于 2017-05-23T19:33:34.120 回答