前言
什么是Lua?
Lua 是一個小巧的腳本語言,巴西里約熱內(nèi)盧天主教大學里的一個研究小組于1993年開發(fā),其設(shè)計目的是為了嵌入應用程序中,從而為應用程序提供靈活的擴展和定制功能。Lua由標準C編寫而成,幾乎在所有操作系統(tǒng)和平臺上都可以編譯,運行。一個完整的Lua解釋器不過200k,在目前所有腳本引擎中,Lua的速度是最快的。這一切都決定了Lua是作為嵌入式腳本的最佳選擇。相比Python和Per的內(nèi)核,Lua的內(nèi)核小于120KB,而Python的內(nèi)核大約860KB,Perl的內(nèi)核大約1.1MB。Lua語言支持面向?qū)ο缶幊毯秃瘮?shù)式編程,它提供了一個通用類型的表table,可以實現(xiàn)數(shù)組、哈希表、集合、對象的功能。Lua支持協(xié)同進程機制。作為一門可擴展的語言,Lua提供簡單而穩(wěn)定的交互接口,如Lua和C程序可通過一個堆棧交換數(shù)據(jù),這使得Lua語言可以快速地和其它語言實現(xiàn)整合。
總體來說,Lua語言具備以下優(yōu)點:
(1)語言優(yōu)美、輕巧
(2)性能優(yōu)良、速度快
(3)可擴展性強。
正因為Lua語言具備了這樣的特點,使得它能和游戲開發(fā)領(lǐng)域的需求完美地結(jié)合起來,因為我們需要這樣的一門語言,它能夠和C/C++進行完美地交互,因為我們需要它對底層進行封裝。它需要足夠地簡單,因為我們需要簡單、靈活、快速地編寫代碼。那么顯然Lua就是我們一直在尋找地這種語言。
目前大部分游戲都采用了Lua語言進行功能開發(fā),在進行多語種發(fā)行的時候就會遇到時區(qū)顯示的問題。以韓國版本為例,場景如下:
1、服務(wù)器處于固定的位置,比如放在首爾機房;
2、玩家所處的位置不確定,可能在韓國,或者是出差在其它國家或地區(qū);
需求:
無論在哪個國家或地區(qū),統(tǒng)一顯示服務(wù)器的當前時間。在PC上查看,即便在國內(nèi)測試的時候也顯示韓國首爾的時間(比北京時間快1個小時)。
實現(xiàn):
-- 北京時間
local serverTime = 1536722753 -- 2018/09/12 11:25
function getTimeZone()
local now = os.time()
return os.difftime(now, os.time(os.date("!*t", now)))
end
-- 8 hour * 3600 seconds = 28800 seconds
local timeZone = getTimeZone()/ 3600
print("timeZone : " .. timeZone)
local timeInterval = os.time(os.date("!*t", serverTime)) + timeZone * 3600 + (os.date("*t", time).isdst and -1 or 0) * 3600
local timeTable = os.date("*t", timeInterval)
--[[
for k, v in pairs(timeTable) do
print(k .. ":" .. tostring(v))
end
]]
print(timeTable.year .. "/" .. timeTable.month .. "/" .. timeTable.day .. " " .. timeTable.hour .. ":" .. timeTable.min .. ":" .. timeTable.sec)
關(guān)注是這個方法: os.date("!*t", now),其中以!為關(guān)鍵。
lua 源碼, loslib.c Line 283 行
static int os_date (lua_State *L) {
size_t slen;
const char *s = luaL_optlstring(L, 1, "%c", slen);
time_t t = luaL_opt(L, l_checktime, 2, time(NULL));
const char *se = s + slen; /* 's' end */
struct tm tmr, *stm;
if (*s == '!') { /* UTC? */
stm = l_gmtime(t, tmr);
s++; /* skip '!' */
}
else
stm = l_localtime(t, tmr);
if (stm == NULL) /* invalid date? */
luaL_error(L, "time result cannot be represented in this installation");
if (strcmp(s, "*t") == 0) {
lua_createtable(L, 0, 9); /* 9 = number of fields */
setallfields(L, stm);
}
else {
char cc[4]; /* buffer for individual conversion specifiers */
luaL_Buffer b;
cc[0] = '%';
luaL_buffinit(L, b);
while (s se) {
if (*s != '%') /* not a conversion specifier? */
luaL_addchar(b, *s++);
else {
size_t reslen;
char *buff = luaL_prepbuffsize(b, SIZETIMEFMT);
s++; /* skip '%' */
s = checkoption(L, s, se - s, cc + 1); /* copy specifier to 'cc' */
reslen = strftime(buff, SIZETIMEFMT, cc, stm);
luaL_addsize(b, reslen);
}
}
luaL_pushresult(b);
}
return 1;
}
從源碼可以看到 ! 調(diào)用了
#define l_gmtime(t,r) gmtime_r(t,r)
gmtime_r 函數(shù)是標準的POSIX函數(shù),它是線程安全的,將日歷時間轉(zhuǎn)換為用UTC時間表示的時間。
注:UTC —— 協(xié)調(diào)世界時,又稱世界統(tǒng)一時間、世界標準時間
也就是說 “!*t” 得到的是一個 UTC 時間,為0度的經(jīng)線(子午線),亦稱本初子午線,通常將它與GMT視作等同(但是UTC更為科學和精確)。

首爾位于東9區(qū),所以實際的時間應該是 UTC + 9,9就是時區(qū)差 —— 9個小時。北京位于東8區(qū),即 UTC + 8。
如何保證游戲內(nèi)全部統(tǒng)一為服務(wù)器的時間呢?
服務(wù)器需要返回給客戶端當前的時區(qū)的差值,比如韓國就返回 9,國內(nèi)就返回 8,越南返回 7,北美返回 –16,記為 serverTimeZone。
服務(wù)端返回當前服務(wù)器時間serverTime(即首爾當前時間),我們只需要將服務(wù)器時間轉(zhuǎn)為 UTC 的時間,然后再加上 serverTimeZone即可。
os.time(os.date("!*t", serverTime)) + serverTimeZone * 3600
這樣無論在哪個地區(qū)或國家,都將顯示首爾的時候,與服務(wù)器顯示的時間就同步上了。
為什么要一直顯示服務(wù)器的時間呢?
游戲中有很多功能是有時間限制的,比如運營活動,或者功能開啟。如果用本地時間就不好控制,統(tǒng)一用服務(wù)器時間避免了很多問題。
可是也容易遇到一個坑,運營配置的活動時間都是針對當前服務(wù)器的時間,例如某個活動的截止時間是:2018-10-08 00:00:00,游戲需要顯示活動截止倒計時。
通常的做法: ployEndTime – serverTime,得到一個秒數(shù),然后將秒轉(zhuǎn)成:xx天xx小時xx分xx秒
serverTime 是固定的,可是ployEndTime就容易出錯,為什么?
serverTime 是在東9區(qū) —— 首爾的時間,而 os.time({year=…}) 是根據(jù)本地時間來算時間的,這中間就存在問題。有一個時差的問題,之前計算一直用的是serverTimeZone —— 一個固定值,而我當前處于地區(qū)或國家,它相對于UTC的時區(qū)不確定的,怎么辦?
用 (currTimeZone – serverTimeZone) * 3600 / 秒,os.time()之后再加上這個時區(qū)差就是首爾當前的時間戳了。國內(nèi)東8 - 東9 = -1,也就是要減去一個1時區(qū),最終將得到首爾地區(qū)的時間戳,再減去 serverTime 就是剩下的秒數(shù)了,然后將它轉(zhuǎn)為 xx 天 xx 小時 xx 分 xx 秒。
最后小結(jié)一下:
1)os.time({year=xx}),這個時間算出來的是針對當前所處時區(qū)的那個時間戳。
2)os.date(“!*t”, 時間戳) 得到的是UTC(時區(qū)為0)的時間戳。
3)獲取當前時區(qū)的值,可以通過文章開頭的 getTimeZone 方法
4)想顯示固定時區(qū)的時間(例如無論在哪都顯示服務(wù)器的時間),只需要將(服務(wù)器)時間戳(秒),通過第2步的方法,得到 UTC 再加上固定的時區(qū)差
5)計算倒計時的時候,需要考慮到 os.time 是取當前時區(qū),需要再將當前時區(qū)減去目標時區(qū),再計劃時間戳
6)夏令時,本身已經(jīng)撥快了一個小時,當需要顯示為固定時區(qū)的時間,則需要減去一個小時
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
您可能感興趣的文章:- Lua在各個操作系統(tǒng)中的開發(fā)環(huán)境配置教程
- lua開發(fā)中實現(xiàn)MVC框架的簡單應用
- 安裝Nginx+Lua開發(fā)環(huán)境
- Centos7 安裝Nginx整合Lua的示例代碼
- Nginx安裝lua-nginx-module模塊的方法步驟
- cocos2dx+lua實現(xiàn)橡皮擦功能
- Lua中三種循環(huán)語句的使用講解
- Lua中的變量與賦值方法
- Lua協(xié)同程序coroutine的簡介及優(yōu)缺點
- Luvit像Node.js一樣寫Lua應用