原文地址:http://www.gameres.com/478430.htmlhtml
實時動做遊戲在近年來獲得迅猛的發展。而遊戲同步問題,成爲你們繼續解決的核心問題之一。早在 2004年,國內遊戲開發還處於慢節奏 RPG滿天飛的狀況下,我就開始實時動做遊戲研究。分別在 2005-2006期間寫了一系列相關文章,被好多網站轉載:算法
幀間同步模式:《幀鎖定同步算法》(2007):
http://www.skywind.me/blog/archives/131
玩法規避模式:《網絡遊戲同步法則》(2005):
http://www.skywind.me/blog/archives/112
預測插值模式:《影子跟隨算法》(2007):
http://www.skywind.me/blog/archives/1145
現在十年過去,網上愈來愈多的人開始討論遊戲同步技術了,然而不少文章每每只針對某種特定的遊戲狀況,而觀點又常常以偏概全。不少人並無真正開發過實時動做遊戲,更別說瞭解同步技術的前世此生了。轉載別人的觀點並加上本身理解的人不少,實際動過手的人不多。避免給更多人形成無謂的誤導,我今天基於先前的實踐和對歐美動做遊戲,戰網遊戲,主機遊戲(PSN,XBox Live等)網絡技術的瞭解,來對這個問題作一個簡單總結:
網速的變化
開發快速動做遊戲,首先要對公網的網絡質量數據有詳細的瞭解。這裏所說到的網速,是指 RTT,數據往返一週的毫秒時間,而非每秒傳送多少 KB/s。我寫這篇文章是基於我 2005-2006年開發的東西來講的,當時國內公網質量比國外差不少:
上圖爲 2005-2006年國內的網絡環境,某三個省級 IDC的狀況採樣。當時公網 RTT平均值基本在100ms,120ms左右徘徊。因此我文中引用了不少 100ms。這個狀況在2009 年之後已經好了不少(60ms的rtt)。到了2012年之後,公網平均 RTT已經下降到平均 40ms-50ms,省內平均10ms之內了:
上圖爲 2015年某省級 IDC的全國延遲狀況,如若全國多布點以及區別電信聯通的話,平均延遲能控制在20ms之內,延遲基本接近國外水平(固然帶寬還差不少),比我當年文章中提到的網絡狀況好了很多。
幀間同步法
關於幀間同步的「
幀鎖定算法
」系列的方法有不少相似實現(包括後面提到的幀間無等待改進,包括 LockStep等),可是他們的核心都是一個:
保證全部客戶端每幀的輸入都同樣
。這樣的方式被格鬥遊戲,RTS和足球(FIFA類)、籃球(NBA)等體育和動做遊戲大量使用,好比咱們熟悉的各大戰網平臺遊戲(Xbox Live等),還有不少基於模擬器的街機對戰平臺。以及很多大型多人橫版動做遊戲。以開發便利,同步邏輯直觀而受到你們歡迎。
幀鎖定算法多用在 C/S模型中(或者一人作主多人作從的P2P裏),它和 LockStep(多用於P2P)共同存在的問題就是 「網速慢的玩家會卡到網速快的玩家」,老式遊戲常常一個角色斷網,全部人就在那裏等待。爲此出現了幀鎖定的改良版本 「樂觀幀鎖定」(具體描述見幀鎖定文章的下半部分)通過了很多遊戲的實踐檢驗。先前還有幾款上線的橫版格鬥頁遊(如熟知的街機三國)用 Flash 的 TCP without NODELAY 來每秒20個關鍵幀的模式(特地找該遊戲開發者確認了一下)跑該算法(因爲近兩年國內網速提升,Flash的 Tcp without NODELAY也能作不少事情了),效果還不錯。
具體實施時用不着按照文所述每個步奏都相同,能夠有不少變通。好比不必定是有變化的時候才通知服務端,有線上某橫版格鬥頁遊就是也能夠每秒 20次向服務端直接發送數據(flash時鐘不許須要本身獨立計時),服務端再每秒 40次更新回全部客戶端,看具體狀況而定。
也有使用 UDP的端遊,客戶端每秒鐘上傳50次鍵盤信息到服務端,丟了就丟了,後面持續發送過來的鍵盤數據會覆蓋前面的數據,因此丟了不要緊,更快捷。固然,UDP也不是必須的,近兩年網速提升很快,省內都能作到10ms的 RTT 了,跨省也就 50ms的rtt,很多頁游上用該方法上裸的 TCP 照樣跑的很順暢。
而近兩年國外動做遊戲領域也涌現出其餘一些新的改良方法,好比 Time Warp,以客戶端先行+邏輯不一致時回滾的方式,帶來了更好的同步效果,俗稱時間回退法。不果國內暫時沒看到有遊戲這麼嘗試,更多的是國外近兩年的雙人動做遊戲比較多,要求遊戲每幀狀態均可以保存,邏輯上開發會複雜一些。國內大部分是超過兩人出去副本的,在3-4人出去 PK的狀況下,引入狀態回退,會讓整個效果大打折扣。不過2人的效果確實有所改進,有興趣的同窗能夠搜索 Time Warp相關的論文。
2009年,雲遊戲(遊戲遠程渲染)技術獲得普遍應用,客戶端上傳操做,服務端遠程渲染,並以低延遲視頻編碼流的方式傳回給客戶端,用的就是這樣相似的技術。客戶端不須要高額的硬件,也不存在盜版問題,其中 Gaikai和 OnLive兩家公司作的比較好。
2012年,Sony推出 Playstation Now技術,能夠在 PSV和 PS3/PS4上玩雲遊戲,玩家不須要購買遊戲就能夠免費體驗必定時間。使得 PSV/PS3等低端硬件也能夠流暢的跑 PS4遊戲。
可是目前國外網絡環境下跑的還比較流暢,國內的網絡環境要低延遲傳送 HD畫質的視頻流還比較困難,視頻都是比較費帶寬的。可是幀鎖定等保證每幀輸入一致的算法,在當今的網絡質量下傳遞一下玩家操做,仍是沒有任何問題的。
狀態同步法
對於邏輯不須要精確到幀的遊戲類型而言(RPG/ARPG,FPS,賽車),容許每一個客戶端屏幕上顯示的內容不一樣,只要將他們統一到一個邏輯中便可,這部分見:「
網絡遊戲同步法則
」(最好給策劃看看這篇,從玩法上規避)。若是是 RPG遊戲,其實更可能是使用障眼法從玩法和動畫效果上減小 「一次性的」,「決定性」的事件便可:
RPG 遊戲的移動很簡單,只須要「誰在哪裏朝着哪裏移動」,客戶端再作一些簡單的平滑處理便可,不須要額外的「時間」參數。好比《魔獸世界》移動時,就是差很少每秒發送一次(座標,朝向,速度),別的客戶端收到之後就會矯正一下,若是矯正錯誤,好比 A原本往北走忽然拐彎向東,這個數據包傳到B上,B屏幕上的A可能在拐彎前往北跑了更遠,導致拐彎向東時被樹卡住,那麼B就會看到A被樹卡了兩秒沒法移動,而後忽然瞬間移動到新的座標,繼續朝着東跑。
一般 RPG攻擊分爲「有鎖定攻擊」和「無鎖定攻擊」,有鎖定攻擊意思是,我朝你發射火球,無論你怎麼跑,火球都會追蹤並射擊到你,好比你在我面前橫着跑過,我向你發射火球,能夠發現火球並非直線飛行,而是曲線追蹤着你就過去了,這叫有鎖定攻擊。無鎖定攻擊通常是範圍攻擊,先播放個動畫(好比揮刀),而後將攻擊請求提交服務器,服務器結果回來時,動畫恰好播放完畢,而後你們一塊兒減血。
而 FPS和 賽車類遊戲的同步性要求比 RPG高不少,每秒發包量也會多不少(10-30個),多半採用位置預測及座標差值的「導航推測算法(DR)」,具體實現見個人:「
影子跟隨算法
」(DR算法的一個改進實現)。
這類算法因爲位置斷定更爲精確,因此計算量大,不少無法服務端判斷,而是客戶端直接判斷,好比 FPS射擊是否打到別人,客戶端先判斷,除了狙擊這種一槍斃命的射擊外基本都是客戶端判斷的。因爲計算更爲複雜,每秒同步發包差很少到 30個以上,這樣的模式下,每局遊戲的人數也不可能不少,通常16人左右。並且不少才用 P2P的方式運行,具體 FPS遊戲的實現,及 DR算法的代碼編寫,見 「
影子跟隨算法
」這篇文章。
其實狀態同步是一種樂觀的同步方法,認爲你們屏幕上的東西不一樣不要緊,只要每次操做的結果相同便可,不須要象「幀間同步」那樣保證每幀都同樣,所以,對網速的要求也沒有 「幀間同步」系列算法那麼苛刻,通常100ms-200ms都是可以接受的(DiabloIII裏面300ms的延遲照樣打),偶爾網絡抖一下,出現1秒的延遲,也能掩蓋過去。然而比起 「幀間同步」,狀態同步方式對玩法有很多要求,諸如 「一次性」,「決定性」的事件要少不少,並且代碼編寫會複雜一些,不果因爲能容忍更壞的網絡狀況,以及容納更多同時遊戲的人數,在一些玩法肯定的遊戲中(RPG,FPS,賽車),被普遍使用。
而狀態同步又分爲「DR同步」和「非DR同步」,前者針對 FPS,賽車或者更激烈點的 ARPG,後者針對 RPG和普通 ARPG。他們對網速的要求和錯誤的容忍度也是不同,固然,帶來的遊戲即時感也是不一樣的。
總得來講,你但願遊戲體驗更爽快,即時感更強,那麼你每秒發包數就越多,每局(副本)支持的人數越少;而你若是追求對網絡的容忍,想下降發包數,而且增長同時遊戲的人數,那麼相應的就須要以下降即時感爲代價,其兩者不可得兼。然而聰明的策劃和程序們總能想出不少好主意,利用障眼法和玩法規避,動做掩蓋等方法,在相同的狀況下來掩蓋延遲,讓玩家「看起來」更加「即時」和「爽快」,而這個方法具體該怎麼作,並無統一的作法,就得你們結合本身的遊戲和玩法,發揮本身的聰明才智了。
結果同步法
結果同步每每比較簡單,位置即便所有錯亂或者延遲好久都沒有關係,由於遊戲過程徹底不在意位置,只在意最後的結果,好比《夢幻西遊》這樣的「回合制 RPG」 遊戲,屏幕上的人走到哪裏確實無所謂,全部操做都是要點擊或者選擇菜單來下命令,象這樣的遊戲背後實際上是文字遊戲,只是加了一個圖形的殼。
遊戲表面上看起來是動做/RTS 遊戲,可是沒有玩家直接協做和對抗,都是單機遊戲,並不須要同步什麼東西,服務端只要監測下結果不離譜便可,延遲檢測都不要緊。基本是 PVE,並且無協做。即便是 PVP也就是打一下別人的離線數據,和無同步回合制遊戲並沒有本質上的區別。
傳輸協議選擇
老話題 TCP仍是 UDP,答案是大部分時候,TCP打開 NODELAY便可,如今網絡狀況好了不少,不必引入新的複雜度。即使是「幀鎖定算法」上線的多人實時格鬥遊戲,也有在用 TCP跑着的。幀間同步若是可以作到更好的架設機房,那麼延遲基本能控制在 10ms之內,將遊戲玩家按照區域分服務器,讓他們選擇更快的服務器。
即使是帶 DR的狀態同步,不少也都是 TCP的,《魔獸世界》和《暗黑破壞神3》都是基於 TCP來實現的,因此個人建議是,先上 TCP,把你的遊戲發佈出去。
固然,等到你的遊戲發佈出去了,開始掙錢了,你想改進你的遊戲效果,特別是高峯期的卡頓比例(須要收集客戶端統計),那麼你可使用 UDP來改進,《街霸4》和《英雄聯盟》都是使用 UDP的,好比你可使用 libenet(英雄聯盟用的)。不過 libenet所採用的傳輸技術,是上世紀的標準 ARQ作法了,《街霸4》所採用的傳輸技術遠遠高過 libenet,若是你想採用更爲現代的傳輸技術,贏得更低延遲的話,可使用個人「快速傳輸協議-KCP」(
http://www.skywind.me/blog/archives/1048
),被再若干上線項目和開源項目使用的協議,效果遠遠 PK libenet。
在使用 KCP時,你能夠用在你 TCP的基礎上,再登錄時服務端返回 UDP端口和密鑰,客戶端經過 TCP收到之後,向服務端的 UDP端口每隔一秒重複發送包含握手信息,直到服務端返回成功或者失敗。服務端經過 UDP傳上來的密鑰得知該客戶端 sockaddr對應的 TCP鏈接,這樣就創建 TCP鏈接到 UDP鏈接的映射關係。爲了保持鏈接和 NAT出口映射,客戶端通常須要每 60秒就發送一個 UDP心跳,服務端收到後回覆客戶端,再在這個 UDP鏈接的基礎上增長調用 KCP的邏輯,實現快速可靠傳輸,這樣一套 TCP/UDP兩用的傳輸系統就創建了。
中國的網絡狀況比較特殊,會存在有些網絡 UDP鏈接不上的狀況,所以都是先鏈接 TCP,而後試圖 UDP,UDP不通的狀況下,退回 TCP也能正常遊戲,一旦 TCP斷開,則認爲 UDP也斷開了。
不果歸根結底,仍是先上 TCP,再根據本身遊戲的特色和是否出現傳輸問題,選擇 UDP。
話題總結
根據遊戲類型,選擇恰當的同步方式和傳輸協議是最基礎的問題,不少講述網絡同步的文章通常就是隻會強調上述那麼多種算法的其中一種方式,好像使用該方式就能夠 hold住全部遊戲同樣的,其實並不是如此。技術須要多和策劃溝通,別策劃一個需求下來,技術就來一句:沒法實現,這樣的遊戲永遠沒有競爭力。就像國內當時都是慢節奏 RPG,偶爾有點 ARPG的時候,你們以爲《DNF》這樣的遊戲沒法實現,因而韓國實現了,在市場上取得了先機,國內才慢慢跟進,再一看,哇塞,好多坑呢。
而後開發者開始在網上尋找各類同步算法,東一榔頭西一棒子,明明是一款須要幀間同步的格鬥遊戲,結果卻上了導航推測,最後發現問題永遠解決不了,一堆 BUG這就叫誤導。正由於我 2004年就開始弄同步相關的問題,期間也指導過很多遊戲設計他們的同步方案,因此此次至關於將之前的觀點作一個總結和點評,根據本身的遊戲類型選擇最適合的同步算法,爲玩家提供更好的體驗纔是關鍵。