文字部分转自: http://1234n.com/?post/qou3eb

supervisor的子进程

一开始使用supervisor的时候,我用的是init/1返回子进程规格列表的方式,并且所有子进程只有两种类型,一种是supervisor进程,一种是gen_server。

但这次代码重构中,我遇到一个情况。如果我可以启动普通的进程而不是gen_server,我就可以把一些我觉得没必要做成gen_server的模块代码精简掉。因为一些模块代码完全没用到任何gen_server的优势,只是借gen_server来做为supervisor的子进程启动。于是我便开始实验如何在supervisor中启动普通的进程。

通过实验,我发现我原先对supervisor的理解完全是没有根据的猜测,果然实践才是检验真理的唯一标准。

首先,我原以为supervisor的子进程规格中提供的Module、Function、Arguments就是子进程的入口,supervisor会通过它来自己启动子进程。但实际上,子进程规格提供的是子进程的启动函数入口,supervisor通过调用这个入口函数来启动子进程,而这个函数通过返回{ok, Pid}来告诉supervisor子进程创建成功。

其次,并不是简单的通过spawn在子进程启动函数中启动一个进程然后返回{ok, Pid}就可以让子进程拥有出错自动重启的功能。实际上,需要使用proc_lib:spawn_link或者proc_lib:start_link启动子进程,才能在子进程出错退出时让supervisor自动重启它。

上面的第二点,我是通过阅读gen_server的代码才了解到的,因为一开始我用来做实验的代码没有用spawn(因为第一点的错误理解),但这个问题很容易就发现了,接着我用了spawn看似正确启动了子进程,但实际上这些子进程一旦出错退出就不会再被启动。我换gen_server实验了一遍,确信用gen_server是可以重启的。于是我便阅读gen_server:start_link的代码一探究竟,从源码中可以看到gen_server模块调用gen模块来启动进程,而gen模块最终通过proc_lib:start_link来启动进程。

proc_lib:start_link和proc_lib:spawn_link的不同之处在于,前者是同步创建子进程,后者是异步创建子进程,proc_lib:start_link调用后会阻塞,直到子进程初始化完毕,调用proc_lib:init_ack后才返回。而proc_lib:spawn_link一调用就会立即返回子进程ID。

测试代码(其中有和主题无关的测试代码,请大家无视):

parent_sup.erl

-module(parent_sup).
-behaviour(supervisor). -export([start_link/0]).
-export([init/1]). start_link() ->
supervisor:start_link(?MODULE, []). init(Args) ->
{ok, {{one_for_one, 1, 60},
[%%{child, {child, start_link, ["hello"]},
%% permanent, brutal_kill, worker, [child]},
{parent, {parent, start_link, ["args"]},
permanent, brutal_kill, worker, [parent]}
]}}.

parent.erl

-module(parent).
-export([start_link/1]).
-export([start_loop/1]). start_link(Args) ->
io:format("~p~n", [Args]),
Ret = proc_lib:start_link(?MODULE, start_loop, [self()]),
io:format("~p~n", [Ret]),
Ret. start_loop(Parent) ->
proc_lib:init_ack(Parent, {ok, self()}),
loop(). loop() ->
receive
Args ->
io:format("~p~n", [Args])
end.

测试命令:

erl -boot start_sasl

22> c(parent).
{ok,parent}
23> c(parent_sup).
parent_sup.erl:10: Warning: variable 'Args' is unused
{ok,parent_sup}
24> c(parent).
{ok,parent}
25> parent_sup:start_link().
"args"
{ok,<0.123.0>} =PROGRESS REPORT==== 16-Jul-2013::11:05:24 ===
supervisor: {<0.122.0>,parent_sup}
started: [{pid,<0.123.0>},
{name,parent},
{mfargs,{parent,start_link,["args"]}},
{restart_type,permanent},
{shutdown,brutal_kill},
{child_type,worker}]
{ok,<0.122.0>}
26> exit(pid(0,123,0), some_reason). =SUPERVISOR REPORT==== 16-Jul-2013::11:06:28 ===
Supervisor: {<0.122.0>,parent_sup}
Context: child_terminated
Reason: some_reason
Offender: [{pid,<0.123.0>},
{name,parent},
{mfargs,{parent,start_link,["args"]}},
{restart_type,permanent},
{shutdown,brutal_kill},
{child_type,worker}] =SUPERVISOR REPORT==== 16-Jul-2013::11:06:28 ===
Supervisor: {<0.122.0>,parent_sup}
Context: shutdown
Reason: reached_max_restart_intensity
Offender: [{pid,<0.123.0>},
{name,parent},
{mfargs,{parent,start_link,["args"]}},
{restart_type,permanent},
{shutdown,brutal_kill},
{child_type,worker}] ** exception exit: shutdown
27> c(parent).
{ok,parent}
28> c(parent_sup).
parent_sup.erl:10: Warning: variable 'Args' is unused
{ok,parent_sup}
29> parent_sup:start_link().
"args"
{ok,<0.138.0>} =PROGRESS REPORT==== 16-Jul-2013::11:07:19 ===
supervisor: {<0.137.0>,parent_sup}
started: [{pid,<0.138.0>},
{name,parent},
{mfargs,{parent,start_link,["args"]}},
{restart_type,permanent},
{shutdown,brutal_kill},
{child_type,worker}]
{ok,<0.137.0>}
30> exit(pid(0,138,0), some_reason).
"args" =SUPERVISOR REPORT==== 16-Jul-2013::11:07:28 ===
Supervisor: {<0.137.0>,parent_sup}
Context: child_terminated
Reason: some_reason
Offender: [{pid,<0.138.0>},
{name,parent},
{mfargs,{parent,start_link,["args"]}},
{restart_type,permanent},
{shutdown,brutal_kill},
{child_type,worker}] {ok,<0.140.0>} =PROGRESS REPORT==== 16-Jul-2013::11:07:28 ===
supervisor: {<0.137.0>,parent_sup}
started: [{pid,<0.140.0>},
{name,parent},
{mfargs,{parent,start_link,["args"]}},
{restart_type,permanent},
{shutdown,brutal_kill},
{child_type,worker}]
true
31>

erlang supervisor中启动普通的进程的更多相关文章

  1. 在Hudson Job中启动daemon进程

    场景 在Hudson中新建一个Job用于构建Web工程,在Job的构建脚本的最后会启动Jetty,观察发现Jetty启动之后一小段时间,进程就终止了.   环境 CentOS 6,Hudson 3.0 ...

  2. jenkin构建项目执行脚本后,脚本中启动的进程也随之关闭的解决办法

    问题描述: 之前用jenkins构建项目(maven项目)后都是通过ssh先将war文件推送到远程服务器,然后执行远程的脚本(更新项目,重启tomcat),一直没有出现问题,今天使用jenkins构建 ...

  3. Erlang虚拟机的启动

    Erlang虚拟机的启动 erl实际上是一个shell脚本,设置几个环境变量之后,调用执行erlexec.erlexec的入口点在 otp_src_R15B01/erts/etc/common/erl ...

  4. [Erlang13]怎么把一个普通的进程挂入Supervisor监控树?

    简单来说:应该是在调用的start_link返回一个{ok,Pid}就可以把这个进程放入监控树Supervisor里面: -module(worker). -author("zhongwen ...

  5. Android系统在新进程中启动自定义服务过程(startService)的原理分析

    在编写Android应用程序时,我们一般将一些计算型的逻辑放在一个独立的进程来处理,这样主进程仍然可以流畅地响应界面事件,提高用户体验.Android系统为我们提供了一个Service类,我们可以实现 ...

  6. Docker学习笔记 - 在运行中的容器内启动新进程

    docker psdoker top dc1 # 容器情况# 在运行中的容器内启动新进程docker exec [-d] [-i] [-t] 容器名 [command] [args]docker ex ...

  7. 在linux中安装jdk以及tomcat并shell脚本关闭启动的进程

    在命令行模式中输入uname -a ,如下图,当界面展示i386就说明本linux系统为32版本,就在官网下载对应jdk版本,或者直接到我的网盘上下载http://pan.baidu.com/s/1c ...

  8. C#/.NET 中启动进程时所使用的 UseShellExecute 设置为 true 和 false 分别代表什么意思?

    原文:C#/.NET 中启动进程时所使用的 UseShellExecute 设置为 true 和 false 分别代表什么意思? 在 .NET 中创建进程时,可以传入 ProcessStartInfo ...

  9. supervisor之启动rabbitmq报错原因

    前言 今天重启了服务器,发现supervisor管理的rabbitmq的进程居然启动失败了,查看日志发现老是报错,记录一下解决的办法. 报错:erlexec:HOME must be set 找了网上 ...

随机推荐

  1. 《Word排版艺术》读后感,兼谈LaTeX

    我有两年多的LaTeX使用经验,用它排实验报告.毕业论文和书籍(半本):Word的使用时间长一些,但真正用好也不过是近一两年的事.这两个软件我都 用得很熟,我想我可以一边谈谈读<Word排版艺术 ...

  2. SQL手工注入小结

    第一步先把IE菜单=>工具=>Internet选项=>高级=>显示友好 HTTP 错误信息前面的勾去掉.否则,不论服务器返回什么错误,IE都只显示为HTTP 500服务器错误, ...

  3. 基于CSRF的XSS攻击

    有些XSS不好利用啊,比如有些后台的XSS漏洞,你进不了别人的后台,怎么能 利用他的XSS漏洞呢?进得了别人的后台,还利用这个XSS漏洞干什么?其实这个时候可以种个后门(如果那是个持久型的XSS,这是 ...

  4. python 如何调用子文件下的模块

    在python开发中,经常会出现调用子文件夹下的py模块 如上图,如果在test.py文件中,要调用meeting文件夹下面的huodongshu.py 模块, 直接在test.py 中 import ...

  5. Lidgren.Network – an introduction to networking in C# games

    Lidgren.Network – an introduction to networking in C# games http://genericgamedev.com/tutorials/lidg ...

  6. 前端性能优化:DocumentFragments或innerHTML取代复杂的元素注入

    来源:GBin1.com 我们的浏览器执行越来越多的特性,并且网络逐渐向移动设备转移,使我们的前端代码更加紧凑,如何优化,就变得越来越重要了.前端给力的地方是可以有 许多种简单的策略和代码习惯让我们可 ...

  7. PHP和数据访问之(插入。删除。和更新数据)

    插入: <?php $conn=@new mysqli('localhost','root','123','mytestdb'); $q_str=<<<EOM insert i ...

  8. mysql 创建函数或者存储过程,定义变量报错

    报错的原因是因为在过程或者函数中存在分隔符 分号(:),而mysql中默认分隔符也是 :,这就导致存储过程分开了 在存储过程外面包一层 delimiter //   code  //就行了

  9. vue 不能检测数组长度 值变化原因解析

    1.vue不能检测数组长度或者值的变化 (1)数组长度变化 未检测到 <!DOCTYPE html> <html lang="en"> <head&g ...

  10. RAC环境下的堵塞(blocking blocked)

    RAC环境下的堵塞不同于单实例情形,由于我们须要考虑到位于不同实例的session.也就是说之前查询的v$session,v$lock对应的应变化为全局范围来查找.本文提供了2个查询脚本,并给出实例演 ...