OTP设计原则学习笔记(1)

 

1、概述

OTP设计原则,是对编写进程、模块、目录等结构化程序代码的一般要求。

1.1 监管树(Supervision Trees)

Erlang/OTP的一个基本概念是监督管理树。它是结构化进程的模型,它的理论基础是“执行人”和“监管人”的监管执行机制。

● 执行人是从事实际运算工作的进程。

● 监管人是监督管理执行人具体工作的进程。发生执行错误,监管人可以命令执行人重新开始工作。

● 监管树把监管人与执行人的关系,组织成分层有序的结构,具有容错机制。

1.2 行为(Behaviours)

监督树中的许多进程,结构相同,模式相同。例如,监督人的结构十分相似,唯一的区别是其监管着不同的子进程。同样,许多执行人的工作是向客户提供服务,或者,充当有限状态机,以及事件处理器譬如书写差错日志。

行为的概念,是对上述工作模式的进一步抽象。目的是为了把进程代码分成2部分,一般属性(行为模块)和具体行动(响应模块)。

行为模块是Erlang/OTP的组成部分。为了实现进程,如监管人,程序员必须实现行动模块,导出预定义的响应函数。

下面的例子,把进程代码分成了2部分:一个简单的服务器,跟踪着许多“通道”。其他进程可以调用函数alloc/0或free/1,分配或释放这些通道。

-module(ch1).

-export([start/0]).

-export([alloc/0, free/1]).

-export([init/0]).

start() ->

    spawn(ch1, init, []).

alloc() ->

    ch1 ! {self(), alloc},

    receive

        {ch1, Res} ->

            Res

    end.

free(Ch) ->

    ch1 ! {free, Ch},

    ok.

init() ->

    register(ch1, self()),

    Chs = channels(),

    loop(Chs).

loop(Chs) ->

    receive

        {From, alloc} ->

            {Ch, Chs2} = alloc(Chs),

            From ! {ch1, Ch},

            loop(Chs2);

        {free, Ch} ->

            Chs2 = free(Ch, Chs),

            loop(Chs2)

    end.

服务器的代码,作为一般属性部分,可以写入文件server.erl: 

-module(server).

-export([start/1]).

-export([call/2, cast/2]).

-export([init/1]).

start(Mod) ->

    spawn(server, init, [Mod]).

call(Name, Req) ->

    Name ! {call, self(), Req},

    receive

        {Name, Res} ->

            Res

    end.

cast(Name, Req) ->

    Name ! {cast, Req},

    ok.

init(Mod) ->

    register(Mod, self()),

    State = Mod:init(),

    loop(Mod, State).

loop(Mod, State) ->

    receive

        {call, From, Req} ->

            {Res, State2} = Mod:handle_call(Req, State),

            From ! {Mod, Res},

            loop(Mod, State2);

        {cast, Req} ->

            State2 = Mod:handle_cast(Req, State),

            loop(Mod, State2)

    end.

第二部分是响应模块 ch2.erl: 

-module(ch2).

-export([start/0]).

-export([alloc/0, free/1]).

-export([init/0, handle_call/2, handle_cast/2]).

start() ->

    server:start(ch2).

alloc() ->

    server:call(ch2, alloc).

free(Ch) ->

    server:cast(ch2, {free, Ch}).

init() ->

    channels().

handle_call(alloc, Chs) ->

    alloc(Chs). % => {Ch,Chs2}

handle_cast({free, Ch}, Chs) ->

    free(Ch, Chs). % => Chs2

注意:

● 上述服务器代码,可用于建设许多不同类型的服务器;

● 例中服务器的名字ch2,对于其服务对象的函数是看不到的,因此,可以改变名字;

● 服务器收发消息的协议,也是调用它的函数不知道的。这是好的编程习惯,它让我们用接口函数改变协议。

● 我们可以扩展server.erl中函数的功能,而不必修改ch2.erl和其他响应模块。

(在ch1.erl和ch2.erl中,有意忽视函数channels/0, alloc/1 和free/2,似乎它们可有可无。其实,完整的函数在后面。这里只是个示例,而实际的程序必须能够处理复杂情况,如分配完成任务的通道,等等。)

channels() ->

   {_Allocated = [], _Free = lists:seq(1,100)}.

alloc({Allocated, [H|T] = _Free}) ->

   {H, {[H|Allocated], T}}.

free(Ch, {Alloc, Free} = Channels) ->

   case lists:member(Ch, Alloc) of

      true ->

         {lists:delete(Ch, Alloc), [Ch|Free]};

      false ->

         Channels

   end.        

不用行为模式的程序代码可能效率更高,但总体代价也大。十分重要的是,要在系统中统一管理全部应用程序。

使用行为模式,有利于程序代码易读易懂。执行效率高的代码,往往难以看懂。

模块server.erl,类似Erlang/OTP 的行为模式 gen_server,但做了极大简化。 

Erlang/OTP的标准行为模式包括:

gen_server 

    实现client-server关系中的服务器;

gen_fsm 

    实现有限状态机;

gen_event 

    实现事件处理功能;

supervisor 

    在监管树中实现监管人。

如果定义了 -behaviour(Behaviour) 而没有定义响应函数,编译器会发出警告:

-module(chs3).

-behaviour(gen_server).

...

3> c(chs3).

./chs3.erl:10: Warning: undefined call-back function handle_call/3

{ok,chs3}

1.3 代码部件applications

Erlang/OTP有许多代码部件,分别实现特定功能。这些部件的术语叫applications,如Mnesia、Debugger。在Erlang/OTP 基础之上建立的最小系统,其applications是Kernel 和 STDLIB。

应用程序的概念有2层含义,程序结构(进程)和目录结构(模块)。

最简单的应用程序没有进程,只有一些函数组成的模块。它就是库程序,例如STDLIB。

应用程序最简单的实现,是用标准的行为模式,做成监管树。

如何具体编程,后面会讲。

1.4 发布的版本

发布的版本,是指由Erlang/OTP的系统子程序发展出来的、结合用户提供的应用程序组成的完整系统。

如何编制发行的版本,后面会讲。

如何在大型环境中安装发布的版本,后面会讲。

1.5 版本管理

版本管理,是指在发布版本之间的升级(更新)和降级(复旧)。具体如何做,后面会讲。

 

你可能感兴趣的:(OTP设计原则学习笔记(1))