日程表服务
skynet 的用户中,问的比较多的一个问题是,为什么我改了系统时间对 skynet 却没有生效?继续追问发现,有这个需求的人大多是想实现一个日程表,到某个特定时间触发特定的任务,修改系统时间是为了测试。
不得不说,通过修改系统时间来测试是个直接、却很糟糕的主意。skynet 的定时器也不依赖系统时间驱动,修改系统时间自然也不会生效。
日程服务是个普遍的需求。在国内网游里,你要不做什么节日任务、每周副本,基本不可能上线。这篇 blog 就谈谈这类需求应该在 skynet 中如何实现。
最好的方法是实现一个单独的服务,用来控制日程。这样、如果你有测试的需求,也可以通过预留的接口让这个服务模拟时间快进,快速触发定时任务了。
通常,日程任务不需要特别高的时间精度,精确到分钟就够了。我们一般不会精确到秒来设定任务开启的时间。日程设置也多半和具体日期、星期几有关。常见的需求类似这样:每周六晚上 8 点开启;每个月的第二个周四中午 12 点;6 月 1 日儿童节活动;等等。
如果要设计一个这样的日程表服务,其实只需要一个订阅接口,类似 skynet timer 那样单次触发的就够了。触发接口就是传入一个时间点,可以描述上面例子中描述的日程。我们在具体活动实现的服务中用 skynet.call 这个日程安排服务,当 skynet.call 返回时,就是日程时间触发点。例如,我们需要每周 6 晚上 8 点固定开启一个活动,可以用一个 while 循环:
while true do
skynet.call(schedule_service, "lua", { wday = 7 , hour = 20 } )
-- 执行活动
end
这个日程表服务每次接到订阅请求,就按照参数计算出下一个符合要求的触发时间点,然后用 skynet.sleep 等到触发时间即可。
如果出于调试需要改动当前时间,直接通知这个服务,调整其内部时间和真实时间的时间差,wakeup 所有正在等待的请求,重新计算等待时间就好了。
这样一个日程表服务并不复杂,我花了一点时间随手实现了一个,供大家参考。它应该有很大的改进空间,明白了其设计之后,应该不难完善它:
local skynet = require "skynet"
local service = require "skynet.service"
local schedule = {}
local service_addr
-- { month=, day=, wday=, hour= , min= }
function schedule.submit(ti)
return skynet.call(service_addr, "lua", ti)
end
function schedule.changetime(ti)
local tmp = {}
for k,v in pairs(ti) do
tmp[k] = v
end
tmp.changetime = true
return skynet.call(service_addr, "lua", tmp)
end
skynet.init(function()
local schedule_service = function()
-- schedule service
local skynet = require "skynet"
local task = { session = 0, difftime = 0 }
local function next_time(now, ti)
local nt = {
year = now.year ,
month = now.month ,
day = now.day,
hour = ti.hour or 0,
min = ti.min or 0,
sec = ti.sec,
}
if ti.wday then
-- set week
assert(ti.day == nil and ti.month == nil)
nt.day = nt.day + ti.wday - now.wday
local t = os.time(nt)
if t < now.time then
nt.day = nt.day + 7
end
else
-- set day, no week day
if ti.day then
nt.day = ti.day
end
if ti.month then
nt.month = ti.month
end
local t = os.time(nt)
if t < now.time then
if ti.month then
nt.year = nt.year + 1 -- next year
else
nt.month = nt.month + 1 -- next month
end
end
end
return os.time(nt)
end
local function changetime(ti)
local ct = math.floor(skynet.time())
local current = os.date("*t", ct)
current.time = ct
if not ti.hour then
ti.hour = current.hour
end
if not ti.min then
ti.min = current.min
end
ti.sec = current.sec
local nt = next_time(current, ti)
skynet.error(string.format("Change time to %s", os.date(nil, nt)))
task.difftime = os.difftime(nt,ct)
for k,v in pairs(task) do
if type(v) == "table" then
skynet.wakeup(v.co)
end
end
skynet.ret()
end
local function submit(_, addr, ti)
if ti.changetime then
return changetime(ti)
end
local session = task.session + 1
task.session = session
repeat
local ct = math.floor(skynet.time()) + task.difftime
local current = os.date("*t", ct)
current.time = ct
local nt = next_time(current, ti)
task[session] = { time = nt, co = coroutine.running(), address = addr }
local diff = os.difftime(nt , ct)
print("sleep", diff)
until skynet.sleep(diff * 100) ~= "BREAK"
task[session] = nil
skynet.ret()
end
skynet.start(function()
skynet.dispatch("lua", submit)
skynet.info_func(function()
local info = {}
for k, v in pairs(task) do
if type(v) == "table" then
table.insert( info, {
time = os.date(nil, v.time),
address = skynet.address(v.address),
})
end
return info
end
end)
end)
-- end of schedule service
end
service_addr = service.new("schedule", schedule_service)
end)
return schedule
用 schedule.submit 可以申请一个时间点,到时后会返回。使用的时候只需要用 skynet.fork 开一个独立线程,循环调用 schedule.submit 即可。
schedule.changetime 支持修改当前时间,供调试用。
Comments
Posted by: marskey | (9) February 7, 2024 10:43 AM
Posted by: Anonymous | (8) September 19, 2017 11:28 AM
Posted by: rinao | (7) September 18, 2017 11:21 AM
Posted by: rinao | (6) September 18, 2017 09:49 AM
Posted by: zhangqiang | (5) September 13, 2017 05:26 PM
Posted by: samuel | (4) September 8, 2017 02:30 PM
Posted by: samuel | (3) September 8, 2017 11:15 AM
Posted by: 发斯蒂芬 | (2) September 7, 2017 04:13 PM
Posted by: x.xiang | (1) September 7, 2017 04:08 PM