1

我有这个正在使用的 gen_server:

-module(user_info_provider).
-export([start_link/0, stop/0]).
-export([init/1, terminate/2, handle_info/2, handle_call/3, handle_cast/2,
 code_change/3]).
-export([request_user_info/2]).

-behaviour(gen_server).

start_link() ->
    gen_server:start_link({local, ?MODULE}, ?MODULE, [], []).

stop() ->
    gen_server:cast(?MODULE, stop).

request_user_info(From,UserId) ->
    gen_server:cast(?MODULE, {request_user_info, From, UserId}).
%% Callback Functions
init(_) ->
    process_flag(trap_exit, true),
    io:format("I am ~w ~n", [self()]),
    {ok, null}.
%% @doc terminate
terminate(Reason, _LoopData) ->
    io:format("Terminating by ~w~n",[Reason]),
    {ok, null}.
handle_cast({request_user_info,From,UserId}, LoopData) ->
    {noreply, LoopData};
handle_cast(stop, LoopData) ->
    {stop, normal, LoopData}.
handle_info(_Info, State) ->
    {ok, State}.
code_change(_OldVsn, State, _Extra) ->
    {ok, State}.
handle_call(_,_From,LoopData) ->
    {ok,ok,LoopData}.

问题是,正如我接下来展示的那样,如果我从 cli 执行它,就像erl -pa ebin/ -s user_info_provider start_link它立即死亡一样,但是我可以从控制台生成它并且它可以工作。

erl -pa ebin -s user_info_provider start_link
Erlang R14B02 (erts-5.8.3) [source] [smp:4:4] [rq:4] [async-threads:0] [kernel-poll:false]

I am <0.32.0> 
Terminating by normal
Eshell V5.8.3  (abort with ^G)
1> user_info_provider:start_link().
I am <0.35.0> 
{ok,<0.35.0>}

如果我不设置process_flag(trap_exit, true)或不直接从控制台启动它,则不会发生这种情况-s module function
我这样启动它是因为真正的 gen_server 要复杂得多,我正在从 Makefile 调用中独立测试它。
有任何想法吗?

4

2 回答 2

5

解决方案是由W55tKQbuRu28Q4xv或不使用建议的解决方案start_link,仅start。这是发生的事情:

-s参数由处理init。粗略地说,init 将spawn使用一个新进程来初始化和运行所有-s参数。之后,这个生成的进程将退出。

由于 init-process 退出,并且您捕获退出,您的进程会收到一条消息{'EXIT', P, Reason},其中Ppid() 是由 init 生成的进程的。此消息由gen_server您的流程部分处理。通常这样的消息会被转发到您的handle_info/2回调(顺便说一句,它在您的代码中有错误的返回值,应该是 noreply)。但在这种情况下,它不会被转发。原因是agen_server包含了它的进程的概念。它通过进程字典和'$ancestors'proc_lib. 现在,如果退出消息来自父级,然后立即调用终止回调并终止进程。这样的消息将不会转发给您。这就是你所看到的。

解决方案,漂亮的解决方案是创建一个小型应用程序,一个主管,并将您的流程置于该主管之下。然后application:start(your_app)从 -s 调用。之所以可行,是因为应用程序控制器是单独运行的。更好的解决方案是构建一个可以自动启动您的应用程序的版本。发布是您的应用程序 + 它的依赖项与 ERTS 运行时捆绑在一起。这样的版本完全独立存在,可以复制到目标主机并运行。因此,目标系统上不需要 Erlang,因为该版本是自包含的。

于 2011-10-13T12:00:49.190 回答
1

使用 supervisor 做最小的应用程序(rebar 可以生成此类应用程序的骨架)并使用 -s 从 cli 运行它。

于 2011-10-12T17:51:38.690 回答