Skynet服务调度

Skynet是多线程框架,其中对应了一些服务(Service),每个服务对应一个Lua虚拟机,一个虚拟机上可以跑多个协程,但同一时刻只能有一个协程,每条消息处理由协程来完成,且运行在保护模式下。Lua层实现的协议池和时序相关的队列,可以类比C++协程相关实现。

Skynet中的协程

Skynet本质上只是一个消息分发器,以服务为单位并给每个服务分配一个独立的ID,可以从任意服务向另一个服务发送消息。在此基础上,在服务中接入Lua虚拟机,并将消息收发的API封装成了Lua模块。

目前使用Lua编写的服务在最底层只有一个入口,就是接收和处理一条Skynet框架转发过来的消息。可以通过skynet.core.callback这个内部用C编写的API(通常由skynet.start调用),把一个Lua函数设置到所属的服务模块中。

每个模块必须设置且只能设置一个回调函数,这个回调函数在每次收到一条消息时,会接收5个参数:消息类型、消息指针、消息长度、消息Session、消息来源。

消息分为两类:

  • 别人对你发起的请求
  • 你过去对外的请求所收到的回应

无论是哪一类,都是通过同一个回调函数进入。在实际使用Skynet时可以直接使用rpc的语法,向外部服务发起一个远程调用,等待对方发送了回应消息后,逻辑接着向下走。那么,框架如何把回调函数的模式转换为阻塞API调用的形式的呢?这多亏了Lua支持协程coroutine,使一段代码运行了一半时挂起,在之后合适的时候再继续运行。

为了实现这点,需要在收到每条请求消息时先创建一个协程,在协程中去运行该类消息的dispatch函数,可使用框架中skynet.dispath函数设置消息的处理函数。之所以必须先创建协程而不能直接调用消息处理函数,是因为无法预知在消息处理的过程中会不会因为阻塞API而需要挂起执行流程。等到第一次需要挂起时才把执行流程绑定到协程上是做不到的。

接着,所有的阻塞都通过coroutine.yield函数挂起当前协程,并把挂起类型以及可能用到的数据传出来。框架会捕获这些参数也就进一步知道去做什么,也也就解释了阻塞API为什么必须在消息处理函数中调用,而不能直接卸载服务的主体代码中的原因。因为初始化部分的代码并不运行在框架创建出来的协程中。

例如:对于skynet.call其实是生成一个对当前服务来说唯一的session号,调用yield给框架发送CALL指令。框架中的resume捕获到CALL之后,会把Session和Coroutine对象记录在表中,然后挂起协程,并结束当前的回调函数。等待Skynet底层框架后续消息进来时再处理。实际上,这里还会处理skynet.fork创建的额外线程。

服务调度API

local skynet = require "skynet"
skynet.sleep(time)

设置当前任务休眠等待的微秒数

skynet.fork(func, ...)

fork用于创建并启动新任务

fork用于启动一个新的任务去执行函数func,实质上它是开了一个协程,函数调用完成后会返回线程句柄。虽然可以使用原生的coroutine.create来创建协程,但这样做会打乱Skynet的工作流程。

skynet.yield()

yield让出当前任务执行流程

yield会让出当前任务执行流程,使本服务内其它任务有机会执行,随后会继续运行。

skynet.wait()

wait让出当前任务执行流程直到使用wakeup唤醒它

skynet.wakeup(co)

wakeup用于唤醒使用waitsleep后处于等待状态的任务

skynet.timeout(time, func)

timeout用于设定一个定时触发函数func,在time * 0.01s后触发。

skynet.starttime()

starttime用于返回当前进程的启动UTC时间(秒)

skynet.now()

now用于返回当前进程启动后经过的时间(微秒)

skynet.time()

time用于通过starttimenow计算出当前UTC时间秒数

sleep 休眠 定时器

skynet.sleep(ti)函数是将当前协程挂起ti个单位时间,一个单位时间是1/100秒。sleep向框架注册注册了一个定时器的实现,框架会在ti时间后发送一个定时器消息来唤醒这个协程。sleep函数是一个阻塞API,返回值nil会告诉你时间到了,返回值BREAK则表示被skynet.wakeup给唤醒了。

$ cd skynet
$ vim demo/service_sleep.lua
local skynet = require "skynet"

skynet.start(function()
  skynet.error("sleep begin")
  skynet.sleep(300)
  skynet.error("sleep end")
end)

$ cp example/config demo/config
$ cp example/config.path demo/config.path
$ vim demo/config.path
# 将example替换为demo
$ vim demo/config
# 将main替换为service_sleep
$ ./skynet demo/config
[:01000009] sleep begin 
service_sleep # 手工输入
[:01000009] sleep end
[:01000002] KILL self

注意:在console服务中输入service_sleep后会发现,新服务不会立即启动,因为console服务正忙于第一个服务的初始化,需要等待3秒后新服务才会被console处理。这种做法实际上是错误的,在skynet.start中服务初始化中是不允许有阻塞的存在,服务初始化要求尽量快的执行完成,所以业务逻辑代码一般不应该写在skynet.start函数中。

fork 在服务中开启新线程

在Skynet的服务中,可以开启一个新的线程用来处理业务,注意这里的线程并非传统意义上的线程,而更像是虚拟线程,其实是通过协程来模拟的。

在Skynet中所有的Lua层函数都是以协程的方式被执行的,包括skynet.fork产生的函数。除非在skynet.start之外调用函数,由于start函数调用timeout产生协程,而fork则产生的是协程列表

Lua层设置的回调函数skynet.dispatch_message主要调用了raw_dispatch_message,这里才是驱动协程函数执行的位置。一个协程结束或挂起后将由suspend函数来接管。

如果入口函数start中没有调用forksleepwait之类的函数,那么驱动start执行的消息将结束。

$ vim demo/service_fork.lua
local skynet = require "skynet"

function task(timeout)
  skynet.error("coroutine fork: ", coroutine.running())
  skynet.error("sleep begin ", timeout)
  skynet.sleep(timeout)
  skynet.error("sleep end")
end

skynet.start(function()
  skynet.error("coroutine start: ", coroutine.running())
  -- 开启新线程来执行task任务
  skynet.fork(task, 500)
end)

$ vim demo/config
start = "service_fork"

$ ./skynet demo/service_fork
[:01000009] coroutine start: thread: 0x7f173b4f1068 false
[:01000009] coroutine fork: thread: 0x7f173b4f1148 false
[:01000009] sleep begin  500
[:01000002] KILL self
[:01000009] sleep end

注意:可以看到当service_fork启动后,console服务仍然可以接收终端输入的服务并启动。若是需要长时间运行并且出现阻塞的情况,可使用skynet.fork创建新的协程来运行。

查看skynet/lualib/skynet.lua文件可知道skynet.fork()函数其实是使用的coroutine.create()函数来实现的。

$ vim lublib/skynet.lua
local coroutine_pool = setmetable({}, {__mode = "kv"})

local function co_create(f)
  local co = table.remove(coroutine_pool)
  if co==nil then
    co = coroutine.create(function(...)
      -- 函数执行完毕
      f(...) 
      while true do
        f = nil
        coroutine_pool[#coroutine_pool + 1] = co
        -- 协程挂起,将由suspend函数接管,执行cmd=="TEXT"分支
        f = coroutine_yield "EXIT" 
        f(coroutine_yield())
      end
    end)
  else
    -- 回到前一次消息挂起的位置返回的f就是要执行的
    coroutine_resume(co, f)
  end
  return co
end

每次使用skynet.fork()其实都是从协程池中获取未被使用的协程,并把该协程加入到fork队列中,等待一个消息调度,然会会一次把fork队列中的协程拿出来执行一遍。执行结束后会把协程重新丢入协程池中,这样可避免重复开启和关闭协程带来的额外开销。

案例:长时间占用执行权限的任务

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

推荐阅读更多精彩内容