该选择哪种函数调用方式?

手册/FAQ (657) 2016-04-14 09:39:29

首先你要知道Module:Func(Args)和Func(Args)的区别在哪里?

如果对细节感兴趣,可以通过这里了解:http://www.cnblogs.com/zhongwencool/p/erlang_hot_code.html

总之:Erlang函数有local call和external call的区别,Local call 就是在函数在被定义的模块里面被调用,可以直接被调用: Func(Args); External call 就是显式的使用Module:Func(Args)来调用或import别的模块进来的调用.

当同一个模块有2个版本被加载里,所有的local call都可以工作在当前版本状态,但是:external call只能调用到最新的版本!

1.使用Fun的情景:

       1.1 先启动1个shell终端,分别名为test1:

erl -name test1

       1.2 test1加载模块mfa_test.erl

 

-module(mfa_fun).

 

-export([cal/2,func/0]).

 

cal(X,Y) ->

     X+Y.

func() ->

       fun(X,Y) -> cal(X,Y) end.

 

1.3在shell中输入:

>Func = mfa_fun:func().

#Fun<mfa_fun.1.52492151>

>Func(1,2).

3

2.使用MFA的场景:

改变mfa_test.erl中cal函数并重新加载这个模块:

cal(X,Y) -> X*Y .

1.5 再运行:

>Func(1,2).

3

可以看到这个Func运行的local版本,如果把func改为以下版本:

func() ->

       fun(X,Y) -> ?MODULE:cal(X,Y) end.

那么结果是永远调用的最新版本代码,有兴趣可以自己验证下,你会感到好神奇,为什么Func变量是没有变化,但得到的结果却发生了变化。。。

 

那么我们可以再进一步想一下:

我们经常使用spawn(Func)来创建一个新的进程,如果我们永远不升级还好,如果热升级,会发生什么情况?

这个local call的func,你怎么也改不了它,所以正常情况下最好使用:

spawn(Mod,FuncName,Args)来创建进程,这样不会影响升级。

再进一步,就是进程里面所有存local call都是无法被升级的,最佳实践就是一直用MFA就好!

THE END