- 浏览: 337204 次
- 性别:
- 来自: 北京
最新评论
-
perfect_control:
真的很详细,一些东西很容易被我忽略掉了
使用fprof进行性能分析 -
leeyisoft:
http://www.erlangqa.com/ 怎么变成 “ ...
Erlang问答网站,欢迎各位提出问题,解答问题。 -
simsunny22:
4年之后我才看到 慢慢的干货
Erlang服务器内存耗尽bug跟踪过程 -
爱死我:
...
使用etop查看系统中进程信息 -
宋兵甲:
在跑这个服务的时候,每秒建立一个客户端连接,连续建立10000 ...
自己写一个tcp 通用服务器
并发和顺序是一个令人纠结的问题。
下面是开发中遇到的一个问题
常规时间,系统表现的很“端庄”,不折腾CPU,不玩弄Mem。可是到高峰时,这个家伙就开始变态了。内存狂飙,直至swap最后无法响应。这个状况,当时折腾了一天多。始终无法找到问题所在。最后通过排查及yufeng的帮助,将问题锁定在某些局部process。
Erlang中默认,所有的Process具有同等的执行机会。
我们的系统中有上万个process处理客户连接,上万个cient process的数据,通过一个data_trans prcess处理。而这个process通过Message将数据发送到其他节点。问题就是这里。
client processes将数据发送给data_trans process后,数据的处理就是一个顺序的过程了,从消息队列中获取一条数据,打包,然后发送到其他Node。显然,在client process增加时,data_trans process的数据处理能力,已经跟不上了。所以导致系统恶化,最终崩溃。
怎么解决呢?
1,可以限定系统的并发连接数,保证服务质量(因为系统某些不足,导致此方法不可行)
2,加大data_trans处理能力,减少瓶颈
至于方法2,也有很多具体的实施方法:将数据打包和数据传输部分进行功能分割;创建多个data_trans组成一个process group
其中创建多个process,对代码改动最少,所以为最终选择。
根据yufeng的建议,实现如下:
使用一个supervisour(simple_one_for_one)管理所有的data_trans进程
data_trans的数目,与scheduler数目一致(8核则数目为8)
每个data_trans name为name_N (N为 1..SchedulerNumber)
调用data_trans时,根据caller,获取当前执行scheduler的X,直接将request跳转到name_X的进程去处理.
好处:
根据scheduler数目创建进程组,减少单个进程处理瓶颈
根据scheduler id直接跳转到进程组中某个进程,减少了中间查询,实现直接映射,效率更高.
简单的示意图(假设系统4核):
caller 1 (scheduler_id 3) -\ /------|- process_1 |\
\/ \
caller 2 (scheduler_id 1) --/ \ /--|- process_2 |--
\ / process supervisor(simple_one_for_one)
caller 3 (scheduler_id 4) -\ / \----|- process_3 |--
\/ /
caller 4 (scheduler_id 2) _/ \____|- process_4 |/
(直接映射)
把这个东西在提升一下,抽象出一个叫gen_server_cter的behaviour,其组装多个子gen_server process,调用时,根据调用者的当前scheduler id映射到对应子process name。
gen_server_cter接口:
start_link(CterName, CbMod, Args)
启动gen_server组
参数:CterName - cter name
CbMod - gen_server callback module
Args - 传递给CbMod的参数
cast(CbMod, Req)
异步调用请求
call(CbMod, Req) ->
同步调用请求
其中CbMod module必须实现一个get_name/1函数,用来实现scheduler id到进程名的映射.
比如(假设CbMod为my_module)
get_name(SchedulerId) ->
list_to_atom(lists:concat([my_module, SchedulerId])).
用法:
gen_server_cter:start_link(my_module_group, my_module, Args)
gen_server_cter:cast(my_module, Req)
gen_server_cter:call(my_module, Req)
就是下面这个module完整代码
-module(gen_server_cter). -behaviour(supervisor). -export([start_link/3]). -export([cast/2, call/2]). %% for supervisor -export([init/1]). -export([behaviour_info/1]). -spec behaviour_info(atom()) -> 'undefined' | [{atom(), byte()}]. behaviour_info(callbacks) -> [{get_name,1}]; behaviour_info(_Other) -> undefined. %% @doc start the server start_link(CterName, CbMod, Args) -> Ret = {ok, _Pid} = supervisor:start_link({local, CterName}, ?MODULE, [{callback, CbMod}, {args, Args}]), %io:format("pid:~p~n", [_Pid]), N = erlang:system_info(schedulers), [{ok, _} = supervisor:start_child(CterName, [{index, I}]) || I <- lists:seq(1, N)], Ret. cast(CbMod, Req) -> Handler = select_handler(CbMod), %io:format("handler is:~p~n", [Handler]), gen_server:cast(Handler, Req). call(CbMod, Req) -> Handler = select_handler(CbMod), gen_server:call(Handler, Req). %% %% supervisor callbacks %% init([{callback, CbMod}, {args, Args} | _]) -> Strategy = {simple_one_for_one, 10, 10}, Mod = {undefined, {CbMod, start_link, Args}, permanent, 3000, worker, [CbMod]}, {ok, {Strategy, [Mod]}}. %% internal API select_handler(CbMod) -> I = erlang:system_info(scheduler_id), CbMod:get_name(I).
update(2009.11.24):
在callback模块中,需要做一些小改动,需要添加一个export函数:
get_name(N :: integer()) -> atom().
返回此server对应的name
还需要修改start_link为:
start_link({index, I}) -> ....
评论
或者在同一服务器上拷贝多个服务
这样的代码改动最小
你的建议没明白问题在哪里
或者在同一服务器上拷贝多个服务
这样的代码改动最小
性能提升了,有个问题还是没解决:处理能力问题应该有适当的流量(请求)控制,否则一个短时的高峰就可能把它挂掉,高峰过后服务也没办法快速恢复。
恩,现在没有有效的流量控制,如果突然出现异常,系统还是会不堪重负
现在只是各部分比较协调.
用法应该是Java代码 gen_server_cter:start_link(my_module_group, my_module, Args) gen_server_cter:cast(my_module, Req) gen_server_cter:call(my_module, Req) gen_server_cter:start_link(my_module_group, my_module, Args)
gen_server_cter:cast(my_module, Req)
gen_server_cter:call(my_module, Req)
而不是Java代码 gen_server_cter:start_link(my_module_group, my_module, Args) gen_server_cter:cast(my_module_group, Req) gen_server_cter:call(my_module_group, Req) gen_server_cter:start_link(my_module_group, my_module, Args)
gen_server_cter:cast(my_module_group, Req)
gen_server_cter:call(my_module_group, Req)
谢谢呵呵。疏忽哈。
gen_server_cter:start_link(my_module_group, my_module, Args) gen_server_cter:cast(my_module, Req) gen_server_cter:call(my_module, Req)
而不是
gen_server_cter:start_link(my_module_group, my_module, Args) gen_server_cter:cast(my_module_group, Req) gen_server_cter:call(my_module_group, Req)
发表评论
-
Erlang问答网站,欢迎各位提出问题,解答问题。
2012-03-18 15:07 5317平时收到很多关于Erlang的问题,我都尽量一一解答,可是时间 ... -
Emakefile并行编译
2011-11-17 13:15 7674项目代码越来越多,使用erlang编译也越来越慢。无论是Mak ... -
Erlang服务器内存耗尽bug跟踪过程
2011-10-25 21:44 21891本文描述朋友Erlang服务器内存耗尽bug的解决过程 ... -
inet:getstat/2小用法
2011-04-27 09:32 4597inet:getstat/2的用处 在 ... -
Erlang游戏开发-协议
2011-04-22 16:10 10740Erlang游戏开发-协议 ... -
Gearman Erlang Client
2010-10-17 21:14 3732Gearman Gearman是一个通用的任务调度框架。 ... -
ECUG归来
2010-10-17 21:02 2991今天ECUG V圆满结束了,不知不觉作为讲师已经参加过3次大会 ... -
gen-erl-app快速生成erlang app 框架
2010-04-07 14:22 4011经常需要创建各种erlang app,这个过程一旦掌握,就很繁 ... -
erl-redis发布
2010-03-30 11:44 5806最近几天因为需要,实现了一个redis erlang clie ... -
用Erlang做了很多事
2010-01-19 14:08 5095因为工作及时间关系,最近比较忙碌,没有太多的时间写文章。 ... -
ecug topic - erlang开发实践
2009-11-11 10:04 3775从ecug归来,感觉不错,大家学习探讨的积极性很高哦。 很高 ... -
reltool用户指南
2009-11-02 22:27 6383说明,最近比较忙,没有太多时间更新blog,请各位朋友谅解. ... -
Erlang定时任务server (仿crontab语法)
2009-09-23 18:03 6388好久不写blog了,看到yufeng老大那么活跃,我也“耐不住 ... -
Erlang进程之错?
2009-07-27 15:06 3705前阵子erlang-china关于erla ... -
CNode指南
2009-07-27 14:13 3356好久不发文章,因为工作太忙。这个东西就凑凑数吧。各位见谅。 ... -
Erlang类型及函数声明规格
2009-06-08 22:41 9578Erlang类型及函数声明 ... -
使用etop查看系统中进程信息
2009-05-29 13:57 6190Erlang提供了丰富的开发工具,你认为没有的时候,很可能是你 ... -
又有人投入Erlang的怀抱了:37Signals Campfire loves Erlang
2009-05-14 23:00 3691就喜欢看这样的东西... This is so juicy ... -
list random shuffle实现
2009-05-07 13:41 4366在项目中需要对list进行随机shuffle,但是在erlan ... -
Erlang开发建议(杂记版)
2009-04-24 18:27 6493以下是在erlang项目开发中的一些记录,即包含很多通俗易懂的 ...
相关推荐
首先,gen_server是一个行为(behaviour),它定义了一组回调函数,这些函数必须由实现gen_server行为的模块实现。这些回调函数包括`init/1`、`handle_call/3`、`handle_cast/2`、`handle_info/2`、`terminate/2`和`...
在Erlang编程语言中,`gen_server`行为是一个强大的模块,用于构建具有状态的、容错的服务。这篇名为“gen_server tasting 之超简单名称服务”的博客文章可能介绍了如何利用`gen_server`来实现一个简单的命名服务。...
《gen_lex_hash_pc:MySQL交叉编译的关键工具详解》 在IT行业中,数据库管理系统是核心组件之一,而MySQL作为开源关系型数据库的代表,广泛应用于各类项目中。在特定环境下,如嵌入式设备或资源有限的PC平台,我们...
标题"Gen_Signature_Android2"指的是一个特定的工具,它用于生成Android应用的签名,这通常是在发布应用到Google Play或其他第三方市场之前所必需的步骤。这个工具可能是为简化开发者的工作流程而设计的,使得他们...
- 处理连接请求,为每个新连接创建一个新的进程(通常是一个gen_server或gen_fsm行为)。 - 注册和登录逻辑,处理用户认证请求。 - 监听和转发消息,确保消息在正确用户间传递。 - 错误处理和异常恢复,确保系统的...
为了实现gen_server,我们需要创建一个持续运行的协程,用于接收和处理消息。我们可以使用`Async.Core.Wakeable`模块来注册我们的协程,使其在接收到新消息时被唤醒。 第一步,定义服务器的状态类型。这将是一个...
总的来说,“Gen_Signature_Android2.zip”提供的工具简化了Android应用MD5签名的获取过程,对于开发者而言,尤其对于那些不熟悉复杂签名流程的用户,这是一个便捷的解决方案。然而,考虑到MD5的安全性问题,它应该...
标题 "srio_response_gen_srio_gen2_0_srio_gen_srio_reponse_SRIO_gen2_SR" 提到的是一个与SRIO(Serial RapidIO)相关的响应生成模块,它可能是一个硬件描述语言(如Verilog或VHDL)设计的源代码文件。SRIO是一种...
Gen_Signature_Android是一个专门用于Android应用签名解析的工具,它可以帮助开发者和安全研究人员深入理解APK文件的签名机制,以及验证APK的签名信息。下面将详细介绍Android应用的签名过程、签名的重要性以及如何...
《Android应用签名详解——以Gen_Signature_Android.apk为例》 在移动应用开发领域,尤其是Android系统中,应用的签名是确保软件安全性和完整性的关键环节。本篇文章将详细探讨Android应用签名的重要性、原理以及...
通用 TCP 服务器 通用 TCP 服务器( gen_tcp_server ) 是一种 Erlang 行为,提供快速简便的方法将 TCP 服务器功能添加到您的应用程序。 它被实现为管理 TCP 连接的主管,因为它是孩子。如何使用它? 运行make来构建。...
在IT行业中,`gen_server` 是Erlang OTP(开放电信平台)框架中的一个核心行为模块,用于构建可靠且容错的服务。它提供了一种模式,使得开发者可以编写并发、状态管理和故障恢复的服务器进程。在"gen_server tasting...
### MKS Gen_L 主板关键知识点解析 #### 一、简介 MKS Gen_L 主板是针对原有 Ramps1.4 开源主板存在的问题而设计的一款优化产品。它结合了 Arduino 2560 和 Ramps1.4 的功能,旨在提供更为稳定且易于使用的解决方案...
这个“华硕Z87主板BIOS_updater_for_4th_Gen_Intel_Core_CPU.zip”压缩包就是用于更新该主板BIOS的工具,以确保与最新硬件和软件的兼容性,解决潜在问题,提高系统的稳定性和性能。 BIOS更新通常包含以下好处: 1. ...
gen_tags.vim, 用来轻松使用 ctags/gtags的vim和neovim的异步插件 gen_tags.vim 为方便用户使用 Vim/ NeoVim,简化了 ctags/ gtags的使用。它用于为你生成和维护多个平台支持的标签,在 Windows/Linux/macOS. 上测试...
在本文中,我们将深入探讨标题为“crc_gen_para_8_ip_crc-gen_verilogIP_crc_”的Verilog HDL实现的CRC校验IP核。这个IP核专门设计用于生成8位参数化的CRC码,适用于多种通讯协议的校验计算。 首先,让我们理解CRC...
《PyPI官网下载:gen_data_model-1.6.1.tar.gz详解》 在Python的开发过程中,包管理器PyPI(Python Package Index)扮演着至关重要的角色,它为开发者提供了一个集中化的地方来发布、查找和安装Python库。本文将...