skynet 笔记续

启动服务

local skynet = require"skynet"

skynet.start(function ()
       -- body
end)

启动一个服务需要调用 skynet 的 start 接口,并向其传入一个启动函数。

定义消息和处理消息

skynet 的服务是用来接收和处理消息的,那么服务会接收什么样的消息呢。通常会在传入的启动函数中注册该服务能够处理的消息类型定义和分发方式。

  1. register_protocol 用于消息定义
  2. dispatch 用于消息分发
local skynet = require"skynet"

local LUACMD = {}

function LUACMD.echo(msg)
    return msg
end

local JSONCMD = {}

function JSONCMD.echo(msg)
    print_r(json.decode(msg))
    return msg
end

skynet.start(function ()

    skynet.dispatch("lua", function(session, source, cmd, ...)
        local f = assert(LUACMD[cmd])
        skynet.ret(skynet.pack(f(...)))
    end)

    -- 定义新的 json 消息类型
    skynet.register_protocol {
        name = "json",
        id = 20,
        pack = function(m)
            return json.encode(m)
        end,
        unpack = function(m)
            local msg_str = skynet.tostring(m)
            return json.decode(msg_str)
        end,
    }

    -- 分发 json 类型的消息
    skynet.dispatch("json", function (session, source, msg)
        local cmd,args = assert(msg.cmd), msg.args
        skynet.ret(cmd(table.unpack(args)))
    end)

end)

以上代码实现了 lua 、json 两种类型消息的分发,其中 lua 是系统已经实现定义好的消息类型, json 类型消息是自定义的类型。

自定义的消息类型时,需提供 name、id、pack、unpack 四个信息。name 和 id 在系统中需唯一,因为系统实现了十多种消息类型占用了前十多个正整数,因而将 id 设为 20。pack 和 unpack 分别是消息的编码和解码函数。

所谓解码,指当收到一个服务收到一个该类型的消息之后,首先会经过 unpack 处理一遍。好比收到的 json
消息实为 json 字符串,需经过 json.decode 成 lua 类型数据以便使用。

类似的方式,当从该服务发送一条 json 消息给其他服务时,也会使用 pack 进行 json.encode 处理一下才会变为 json 格式字符串。当然该其他服务也需实现了能处理 json 消息才好。

实现服务间通信

大多数情况下,仅使用 lua 消息就可以很好的实现我们的业务,因而一个服务的样子大致是这样:

local skynet = require"skynet"

local CMD = {}

function CMD.echo(msg)
    return msg
end

function CMD.foo()
    return "bar"
end

skynet.start(function ()
    skynet.dispatch("lua", function( session, source, cmd, ... )
        local f = assert(CMD[cmd])
        skynet.ret(skynet.pack(f(...)))
    end)
end)

按这个方式,模拟两个 玩家服务 通过 聊天服务 互发消息。

chatd.lua

local skynet = require"skynet"
require"skynet.manager"

local agents = {}

local mt = {}
mt.__index = mt

function mt.msg(from,to,msg)
    local to_agent = assert(agents[to])
    skynet.send(to_agent,"lua","msg",from,msg)
end

function mt.add_agent(name,agent)
    agents[name] = agent
end

skynet.start(function ()

    skynet.dispatch("lua", function(session,source,cmd,...)
        local f = assert(mt[cmd])
        return skynet.ret(skynet.pack(f(...)))
    end)
    skynet.register(".chatd")
end)

agent.lua

local skynet = require"skynet"

local name

local mt = {}
mt.__index = mt

function mt.msg(from, msg)
    skynet.error(string.format("agent [%s] get msg:[%s] from agent[%s]", name,msg,from))
end

function mt.send(from,to,msg)
    local chatd = skynet.localname(".chatd")
    skynet.send(chatd,"lua","msg",from,to,msg)
end

function mt.setname(agent_name)
    name = agent_name
end


skynet.start(function (agent_name)
    name = agent_name

    skynet.dispatch("lua", function(session,source,cmd,...)
        local f = assert(mt[cmd])
        return skynet.ret(skynet.pack(f(...)))
    end)
end)

main.lua

local skynet = require"skynet"

skynet.start(function()
    local chatd = skynet.newservice("chatd") -- 启动聊天服务
    local tom = skynet.newservice("agent")   -- 启动玩家并设置名称为tom
    skynet.call(tom,"lua","setname","tom")
    local jim = skynet.newservice("agent")     -- 启动玩家并设置名称为jim
    skynet.call(jim,"lua","setname","jim")

    skynet.call(chatd,"lua","add_agent","tom",tom) -- 把 tom 注册到 聊天服
    skynet.call(chatd,"lua","add_agent","jim",jim)    -- 把 jim 注册到 聊天服

    skynet.send(tom,"lua","send","tom","jim","foo")  -- 通知 tom 让其向 jim 发消息
    skynet.send(jim,"lua","send","jim","tom","bar")    -- 通知 jim 让其向 tom 发消息
    skynet.exit()
end)

执行

skynet/skynet etc/config

输出

[:00000001] LAUNCH logger 
[:00000002] LAUNCH snlua bootstrap
[:00000003] LAUNCH snlua launcher
[:00000004] LAUNCH snlua cdummy
[:00000005] LAUNCH harbor 0 4
[:00000006] LAUNCH snlua datacenterd
[:00000007] LAUNCH snlua service_mgr
[:00000008] LAUNCH snlua main
[:00000009] LAUNCH snlua chatd
[:0000000a] LAUNCH snlua agent
[:0000000b] LAUNCH snlua agent
[:00000008] KILL self
[:0000000b] agent [jim] get msg:[foo] from agent[tom]
[:00000002] KILL self
[:0000000a] agent [tom] get msg:[bar] from agent[jim]
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,123评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,031评论 2 384
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,723评论 0 345
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,357评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,412评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,760评论 1 289
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,904评论 3 405
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,672评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,118评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,456评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,599评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,264评论 4 328
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,857评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,731评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,956评论 1 264
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,286评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,465评论 2 348

推荐阅读更多精彩内容

  • Lua 5.1 参考手册 by Roberto Ierusalimschy, Luiz Henrique de F...
    苏黎九歌阅读 13,763评论 0 38
  • 1、通过CocoaPods安装项目名称项目信息 AFNetworking网络请求组件 FMDB本地数据库组件 SD...
    阳明先生_X自主阅读 15,969评论 3 119
  • 引言 对于序列化概念,如果是学习过Java的人,相信一定不会陌生,序列化就是将对象的数据、状态转换成能够存储或者传...
    lindx阅读 3,164评论 0 2
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,737评论 25 707
  • 计划在工作上的作用不言而喻。因此,如何在工作计划中设定可执行目标显得异常重要。 每年年初、季初都是设定各项目或个人...
    0372ab113946阅读 701评论 0 2