ms()用途浅析
erlang系统的application,稍微复杂一点的都会提供一个ms/0的导出函数,而且这个导出函数通常在文档里面找不到描述,很奇怪不是吗?
比如mnesia就有这样的ms, 我们来看下:
$ erl Erlang R17A (erts-5.11) [source-b7fbc28] [64-bit] [smp:16:16] [async-threads:10] [hipe] [kernel-poll:false] Eshell V5.11 (abort with ^G) 1> mnesia:ms(). [mnesia,mnesia_backup,mnesia_bup,mnesia_checkpoint, mnesia_checkpoint_sup,mnesia_controller,mnesia_dumper, mnesia_loader,mnesia_frag,mnesia_frag_hash, mnesia_frag_old_hash,mnesia_index,mnesia_kernel_sup, mnesia_late_loader,mnesia_lib,mnesia_log,mnesia_registry, mnesia_schema,mnesia_snmp_hook,mnesia_snmp_sup, mnesia_subscr,mnesia_sup,mnesia_text,mnesia_tm, mnesia_recover,mnesia_locker,mnesia_monitor,mnesia_event] 2>
看起来貌似只是返回组成mnesia的模块列表而已。
那么它的作用是什么呢?
复杂一点的程序都需要在运行期间来进行观察或者优化,比如说dbg跟踪一个函数或者模块运作的时候,是需要这个模块的名字的,如:
dbg:tp(Module,MatchSpec)
那么如果要跟踪整个application的运作,我们通常会写这样的代码:
[do_some_thing(M) || M<-myapp:ms()].
所以这就是ms的意义所在。
现在的问题是mnesia的代码是把ms的模块硬编码的,这样会带来一个维护的问题,比如添加,改名或者减少一个模块都要记得去修改这个列表,很麻烦。
[erlang]
%mnesia.erl
ms() ->
[
mnesia,
mnesia_backup,
mnesia_bup,
...
].
[/erlang]
程序员是个很懒的群体,必定不会这么挫的,于是rebar就专门花了点力气把这个事情做的漂亮。
rebar在编译application的时候,会把src/myapp.app.src的内容添加以下内容:
{modules,[mod_a, mod_b,…]}
形成ebin//myapp.app文件,这个文件是每个app必须的!
rebar处理这个事情的核心代码如下:
%%rebar_otp_app.erl AppVars = load_app_vars(Config1) ++ [{modules, ebin_modules()}], ebin_modules() -> lists:sort([rebar_utils:beam_to_mod("ebin", N) || N <- rebar_utils:beams("ebin")]).
Post Footer automatically generated by wp-posturl plugin for wordpress.
Recent Comments