分类: Erlang2012-08-06 18:55 867人阅读 评论(0) 收藏 举报

首先清楚gen_server提供C/S架构中的服务端的实现,即定义了自己一套规范的服务器框架。

在以上基础上,具体学习下gen_server的实现过程。

先是定义模块的行为模式为gen_server

  1. -module(lqg).
  2. -behaviour(gen_server).
  3. .
  1. -module(lqg).
  2. -behaviour(gen_server).
  3. .
  1. -module(lqg).
  2. -behaviour(gen_server).
  3. .

这里既然用了gen_server框架,那必须要实现gen_server的方法:gen_server callbacks

  1. -export([init/1, handle_call/3, handle_cast/2, handle_info/2,  terminate/2, code_change/3]).
  1. -export([init/1, handle_call/3, handle_cast/2, handle_info/2,  terminate/2, code_change/3]).
  1. -export([init/1, handle_call/3, handle_cast/2, handle_info/2,  terminate/2, code_change/3]).

外加模块的对外调用函数:API

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

接下来就是实现上面定义的函数:

  1. start_link() ->
  2. gen_server:start_link({local, lqg}, lqg, [], []).
  1. start_link() ->
  2. gen_server:start_link({local, lqg}, lqg, [], []).
  1. start_link() ->
  2. gen_server:start_link({local, lqg}, lqg, [], []).

对于start_link,
第一个参数为创建一个名为Name的server,现在的情况为在gen_srever将在本地被注册为lqg
第二个参数lqg, 则是回调模块的名字,也就是回调函数所放的那个模块。在这里,接口函数( start_link)和回调函数(init, handle_call 和 handle_cast)。一般来说这是好的编程实践,将代表同一个进程的代码包含在同一个模块中。
第三个参数[], 这个值将被原封不动传递给回调函数init。在这里,init无须任何输入数据将忽略这个参数.
第四个参数[],是参数的列表。
这里要注意的是:
1.gen_server:start_link 是同步的。只有等到gen_server被完全初始化并准备接受请求之后才会返回
2.如果gen_server是某棵监督树的一部分,即gen_server是由一个督程启动的,那么必须使用
gen_server:start_link 。还有另外一个函数 gen_server:start 用于启动一个独立的gen_server,即不是某棵监督树一部分的一个gen_server

  1. init(_Args) ->
  2. {ok, channels()}.
  1. init(_Args) ->
  2. {ok, channels()}.
  1. init(_Args) ->
  2. {ok, channels()}.

在注册名称成功后,新的gen_server进程会调用回调函数lqg:init([])
init返回{ok, State} ,其中State 是gen_server的内部状态。在这里,状态就是可用的频道channels。

  1. alloc() ->
  2. gen_server:call(lqg,alloc).
  1. alloc() ->
  2. gen_server:call(lqg,alloc).
  1. alloc() ->
  2. gen_server:call(lqg,alloc).

同步请求alloc() 用gen_server:call/2 实现.lqg 是gen_server的名字,必须和启动时的名字一样。alloc 是实际的请求
此时,请求以消息的形式发送给这个gen_server。当收到了请求之后,gen_server调用handle_call(Request,From, State) ,它应返回一个元组 {reply, Reply, State1}。Reply是需要回馈给客户端的答复,同时State1 是gen_server的状态的新值。

  1. handle_call(_Request, _From, State) ->
  2. {Ch, State2} = alloc(State),
  3. {reply, Ch, State2}。
  1. handle_call(_Request, _From, State) ->
  2. {Ch, State2} = alloc(State),
  3. {reply, Ch, State2}。
  1. handle_call(_Request, _From, State) ->
  2. {Ch, State2} = alloc(State),
  3. {reply, Ch, State2}。

在这里,应答是分配了的频道Ch 然后gen_server将等待新的请求,并且现在保持了一个最新的可用频道的列表。

  1. handle_cast({free,Ch},Chs) ->
  2. Chs2 = free(Ch,Chs),
  3. {noreply, Chs2}.
  1. handle_cast({free,Ch},Chs) ->
  2. Chs2 = free(Ch,Chs),
  3. {noreply, Chs2}.
  1. handle_cast({free,Ch},Chs) ->
  2. Chs2 = free(Ch,Chs),
  3. {noreply, Chs2}.

在这里,新的状态便是更新过的可用频道列表Chs2 。gen_server现在又可以接受新的请求了。

  1. free(Ch) ->
  2. gen_server:cast(lqg,{free,Ch}).
  1. free(Ch) ->
  2. gen_server:cast(lqg,{free,Ch}).
  1. free(Ch) ->
  2. gen_server:cast(lqg,{free,Ch}).

异步请求free(ch) 使用 gen_server:cast/2 实现
lqg 是gen_server的名称。{free, Ch} 是实际的请求。
请求被装在一个消息中发给gen_server的cast ,这调用了 free ,然后返回了 ok 。当gen_server收到请求之后,它会调用handle_cast(Request, Stats) ,会返回一个元组{noreply, State1} 。 State1 是gen_server状态的新值。

  1. handle_info(_Info, State) ->
  2. {noreply, State}.
  1. handle_info(_Info, State) ->
  2. {noreply, State}.
  1. handle_info(_Info, State) ->
  2. {noreply, State}.

用来处理请求之外的信息。

  1. terminate(_Reason, _State) ->
  2. ok.
  1. terminate(_Reason, _State) ->
  2. ok.
  1. terminate(_Reason, _State) ->
  2. ok.

终止函数,终止正在运行的进程

若gen_server是某个监督树的一部分,则无需停止函数。它的督程会自动终止它;

如果gen_server并非某个监督树的一部分,那么可以用一个停止函数;
如果在终止之前需要进行一些清理工作,那么关闭策略必须是一个超时值,同时gen_server必须在init 函数中设置为捕获退出信号。当gen_server被要求关闭时,它就会调用回调函数terminate(shutdown, State)。

  1. code_change(_OldVsn, State, _Extra) ->
  2. {ok, State}.
  1. code_change(_OldVsn, State, _Extra) ->
  2. {ok, State}.
  1. code_change(_OldVsn, State, _Extra) ->
  2. {ok, State}.

正如其名,此函数用来进行代码版本替换。是server热部署或代码升级时做callback修改进程状态
_OldVsn:旧版本 State:gen_server的内部状态 _Extra:原封不动的传递过来的更新指令
如果更新成功,返回{ok,State2},如果失败返回{error,Reason},并回滚到旧版本。

OTP gen_server的更多相关文章

  1. erlang OTP gen_server 图解分析

    http://www.hoterran.info/otp-gen_server-sourcecode 在阅读erlang的otp源码gen_server.erl的时候,一直想写点什么,用一种最好的方式 ...

  2. 使用Erlang和Yaws开发REST式的服务

    看过那张很出名的“Apache vs. Yaws”图么?是不是在考虑你也应该使用Yaws了?这些图给人的第一印象是,Yaws在可伸缩性上具有难以置信的巨大优势,它可以扩展到80000个并行的连接,而 ...

  3. Mochiweb的设计分析

    http://blog.csdn.net/dp0304/article/details/6994435 Web服务器的基本工作大致分3步: 接收HTTP请求: 处理HTTP请求,生成响应内容: 发送响 ...

  4. Erlang OTP编程初体验——gen_server和行为模式

    http://blog.sina.com.cn/s/blog_3fe961ae0101k4p6.html 行为模式其实非常类似于面向对象语言中的接口,至少笔者是这么理解的.OTP行为模式将一些反复出现 ...

  5. [Erlang 0119] Erlang OTP 源码阅读指引

      上周Erlang讨论群里面提到lists的++实现,争论大多基于猜测,其实打开代码看一下就都明了.贴出代码截图后有同学问这代码是哪里找的?   "代码去哪里找?",关于Erla ...

  6. Erlang 104 OTP

    笔记系列 Erlang环境和顺序编程Erlang并发编程Erlang分布式编程YawsErlang/OTP 日期              变更说明 2014-12-21 A Outline, 1 A ...

  7. Gen_server行为分析与实践

    1.简介 Gen_server实现了通用服务器client_server原理,几个不同的客户端去分享服务端管理的资源(如图),gen_server提供标准的接口函数和包含追踪功能以及错误报告来实现通用 ...

  8. 如何设置gen_server在退出时执行相关操作

    如果gen_server在监控树中不需要stop函数,gen_server会由其supervisor根据shutdown策略自动终止掉.如果要在进程终止之前执行清理,shutdown策略必须设定一个t ...

  9. Erlang OTP gen_event

    转自:http://www.myexception.cn/program/1569725.html Erlang OTP gen_event (0) 原英文文档:http://www.erlang.o ...

随机推荐

  1. 在Linux(Debian)环境下搭建并运行GPU

    首先通过以下命令查看是否GPU驱动成功: 注意:需要在bash终端输入 import tensorflow as tf hello = tf.constant('Hello, TensorFlow!' ...

  2. jQuery的offset、position、scroll,元素尺寸、对象过滤、查找、文档处理

    jQuery_offset和position var offset = $('.xxx').offset() console.log(offset.left.,offset.top)xxx相对于页面左 ...

  3. 蚂蚁金服ATEC城市峰会上海举行,三大发布迎接金融科技2019

    2019年1月4日,蚂蚁金服ATEC城市峰会以“数字金融新原力(The New Force of Digital Finance)”为主题在上海举办.稠州银行副行长程杰.蚂蚁金服副总裁刘伟光.蚂蚁金服 ...

  4. mysql自增长主键,删除数据后,将主键顺序重新排序

    用数据库的时候,难免会删除数据,会发现设置的主键增长不是按照正常顺序排列,中间有断隔比如这样. 以我这个情况举例 处理方法的原理:删除原有的自增ID,重新建立新的自增ID. ALTER TABLE ` ...

  5. CSS实现水平垂直居中的1010种方式

    转载自:CSS实现水平垂直居中的1010种方式 划重点,这是一道面试必考题,很多面试官都喜欢问这个问题,我就被问过好几次了 要实现上图的效果看似很简单,实则暗藏玄机,本文总结了一下CSS实现水平垂直居 ...

  6. curl的Get请求,封装方法

    //GET请求//参数1是请求的url//参数2是发送的数据的数组//参数3是其他GET选项private function curlGET($url, array $get = array(), a ...

  7. gdb 命令汇总

    https://ftp.gnu.org/old-gnu/Manuals/gdb/html_node/gdb_109.html whatis expr 举例 whatis  uint64      -& ...

  8. app在admin中显示成我们想要的中文名

    在django的开发中,很多时候我们希望app在admin中显示成我们想要的中文名,而不是显示默认的app_label名称. 比如我们有一个blog应用,在我们的blog app目录下面,默认会生成一 ...

  9. inconfont的使用

    iconfont 进入阿里的矢量图标库,进入到我的项目(图标库)中,可以看到如下页面 根据不同的使用方式(Unicode.Font class.Symbol)可将对应文件下载至本地或直接在项目中引入 ...

  10. mysql5.7.25安装

    附:mysql安装包 链接:https://pan.baidu.com/s/1vROdBSw0GiMWCRpuwmqFCg 提取码:ug4o a.运行mysql-installer-community ...