经思考后,crontab 还是废弃好了。
目前决定还是回归 set_times_check 的做法,然后以最近应该优先
完成的自动排程拍卖系统为例,/cmds/std/_blarket.c 里面要加上
times_check 函数,以华丽与污痕书店为例,它要透过下指令触发,
使用者下指令,它才会去 set_times_check,所以我只要透过下指令
的方式设定排程即可,以书店为例,设定的呼叫段如下:
几秒后
times_check->set_times_check(names,shop_files,({str,sk}),sk/50);
所以假设我希望 2/15 晚上 21:00 开启拍卖,这时需要一个计算函
数,计算上面时间与现在时间的秒数差,这个有简单的方式
Sat Jan 13 21:03:10 2018 现在时间
Thr Feb 15 21:00:00 2018 目标时间
年先不用管,上面做 sscanf 取出 mon 及 day:
times=time();
sscanf(ctime(times),"%s %s %s %d:%d:%d %s",week,mon,day,hour,min,sec,year);
tar_mon="Feb";
tar_day="15";
tar_hour=21;
tar_min=0;
t=0;
// 这个循环做完只是一瞬间的事而已
while(mon!=tar_mon || day!=tar_day)
{
t+=86400;
sscanf(ctime(times+t),"%s %s %s %d:%d:%d %s",week,mon,day,
hour,min,sec,year);
}
脱离上面 while 的条件就是 mon==tar_mon && day==tar_day,因为
不比年,所以循环最多跑三百多次。接着:
t=t+(tar_hour-hour)*3600+(tar_min-min)*60; // 秒不用管
几秒后
times_check->set_times_check("blarket","/cmds/std/_blarket",({参数}),t);
我刚有用 running 实验,2849340 秒后就是 2018/2/15 晚上九点:
if(1)
{
string week,mon,day,tar_mon,tar_day,year;
int times,t,hour,min,sec,tar_hour,tar_min;
// 以 2018/02/15 21:00 为例
times=time();
sscanf(ctime(times),"%s %s %s %d:%d:%d %s",week,mon,day,hour,min,sec,year);
tar_mon="Feb";
tar_day="15";
tar_hour=21;
tar_min=0;
t=0;
// 这个循环做完只是一瞬间的事而已
while(mon!=tar_mon || day!=tar_day)
{
t+=86400;
sscanf(ctime(times+t),"%s %s %s %d:%d:%d %s",week,mon,day,
hour,min,sec,year);
}
t=t+(tar_hour-hour)*3600+(tar_min-min)*60;
write("t="+t+", ctime(times+t)="+ctime(times+t)+"\n");
}
========== 程式执行区 ==========
t=2849340, ctime(times+t)=Thu Feb 15 21:00:58 2018
========== 程式执行区 ==========
这比 crontab 稳定多了,因为不需要判断年,就可以很简单的做好
防止循环过大的判断(例如在 while 循环内加上计数器,超过 400
次就代表给的时间有问题就不设定)。
而且,有需要时再设定即可。
就算是 loop 型的 times_check 同样也可以交给目标物件去做,好
处就是自己的目标物件要跑怎么样的 loop 判断,自己写最弹性。
这里还可以导入 boat 的概念:
// 航行计划
set("plane",({ ({"菲里德军港","/u/l/laechan/area/felid/room/285-军港",30,90}),
({"西斯迪克港","/u/l/laechan/area/sisdic/room/001",30,90}),
({"塔塔克高原",TR"015",30,90}),}));
仿照航行计划,自己编拍卖计划
// 拍卖计划
set("blarket",([
"newyear":({ ({......,几秒后,等待几秒,..}),
.
.
}),
"santa": ...
]));
先编好拍卖计划,再透过指令去设定,不同的节日执行不同的拍卖
计划即可,根据我 blarket 的写法
void create()
{
seteuid(getuid(this_object()));
if(file_exists("/open/cmds/blarket.o"))
restore_object("/open/cmds/blarket");
else
data["boards"]=([]);
set("pre_clean",1); // 避免被系统当成真正的 room. 2010/08/28
}
我在这边多写一个读取排程的物件资料即可。
以上应该是比较容易实行的方式,维持 times_check 的泛用性,
wiz 不用习惯新的系统(crontab),只要知道 times_check 的原理
即可,而这东西已经存在很久也有范例了(/std/new_ob/boat.c)。
最近会优先改 blarket,改好后会实验,接着就会改农场控制室,
将星星日改成可由系统去定时跑,例如六日一定会跑,再来可以一
三五跑一次之类的,这个改好后也会实验,再有时间,就是月例大
赛的掷筊王比赛的定期每月月底或每月一号开启,什么时间开启不
重要,因为开启后至少 n 天的时间是线上玩家都可以去掷的。
Laechan