一款已上市MMO手遊地圖同步方案總結

1. 客戶端地圖格子的相關知識緩存


 在2.5D的MMO遊戲裏,角色是經過3D的方式渲染,2D的地圖是經過2D的方式顯示,因此在客戶端通常會有三個座標系:服務器


a) 3D座標系:全部須要3D渲染的角色和光效,都以3D座標系中定位。網絡

b) 2D座標系:用來定位和繪製固定的2D地圖元素,好比草皮、馬路等。性能

c) 3D座標裏的格子座標系:用來實現打掩碼、自動尋路和進行一些座標配置(好比NPC和怪物初始的位置)。使用格子座標,一是爲了方便打掩碼和進行自動尋路的計算(經典的A*尋路),二是爲了更方便查找座標的具體位置。spa


  端遊使用的格子大小通常爲(64, 32),手遊的精確度要求低一些,能夠用(100, 50),即3D座標系里長爲100寬爲50的矩形,便是格子座標系裏的一個座標。示例圖以下:orm

一款已上市MMO手遊地圖同步方案總結

2. 服務器同步大格子:9宮格

對象

  MMO遊戲裏,玩家要能看到地圖上全部角色的行爲,這就須要將其它玩家的動做都經過網絡數據同步過來。同步通常使用9宮格來肯定,哪些玩家的數據要同步過來,而後本身的行爲要同步給哪些玩家。遊戲


  服務器大格子的大小,以3*3的格子要總比客戶端顯示範圍要大一點爲原則。比客戶端大一點,是爲了預留資源加載的時間。進程


  以下圖所示,綠色表示手機客戶端的顯示區域,當角色A在格子6中時,他能夠看到1,2,3,5,6,7,9,10,11這9個格子裏的內容,那麼當他的狀態發生變化時,就須要同步給在這9個格子裏的全部玩家;一樣,當這9個格子裏的有玩家或者怪物的狀態改變時,也須要都同步給角色A。資源


一款已上市MMO手遊地圖同步方案總結


當角色A移動到角色B所在的格子(7),則他將再也不看到1,5,9這三個格子裏的內容(玩家和怪物),同時他將新看到4,8,12這三個格子裏的地圖內容。因此這個過程當中,服務器要下發消息,刪除角色A所在的客戶端裏的1,5,9這三個格子裏的地圖內容,同時下發消息新增4,8,12這三個格子裏的地圖內容(類型一)。


  推薦大格子具體的大小,按客戶端iPhone4S的960*640分辨來制定,取屏幕長寬的1/2大一些,能夠定爲640*360。

3. 角色的移動同步

  地圖上角色的同步能夠分爲位移的同步和行爲(好比放技能)的同步。這裏主要討論位移的同步方式。


  位移同步的目的是爲了將本身的位置變化發給服務器,而後由服務器經過9宮格的方式轉發給周圍的其餘玩家。


  有的端遊是以客戶端格子的基本單位進行同步,當玩家從一個格子移動到了另外一個格子時,就發消息通知給服務器。這種方式的缺點就是:


1、同步的延遲。玩家從一個格子開始移動,移動到另外一個格子後,才發消息給服務器,服務器再轉發給其它客戶端,那其它客戶端的玩家位置,總會有一點延後。

2、當網絡不穩定的時候,很容易看到其它玩家不是均速的移動,好比玩家位置沒動,而後一會兒瞬移到了下一個格子。


  咱們採用的方式,是同步狀態的變化,而後由客戶端來觸發服務器對大格子跨越的判斷:


d) 當玩家點擊地圖上某個地方,或者改變了搖桿方向,玩家的運行狀態就變化了,即向某個座標點移動。狀態變化的時候,客戶端就當即給服務器發消息,而後服務器進行轉發。這樣若是忽略了網絡的延遲,那這個角色在全部客戶端上,幾乎是同時開始移動。


  若是移動過程當中沒有其它變化,則整個移動過程當中只有一次消息同步。這裏須要處理一個問題,就是服務器須要知道這個角色何時跨越了服務器的同步大格子,當角色跨越了同步大格子時,服務器就須要進行第二節裏(類型一)的操做。


e) 如何判斷角色的移動過程當中跨越了同步大格子,有的遊戲裏採用服務器判斷的方式,即根據角色的移動速度和方向,計算出跨越的時刻,而後使用一個Timer來觸發。同時若是服務器要取這個角色的當前位置,則須要經過運動公式來進行計算。這個方案相對精確一些,但比較複雜,服務器也須要爲每個移動的角色設定一個Timer,對服務器的性能有所影響。


  咱們採用的方式,是由客戶端判斷角色每移動一小段距離,而後發消息通知服務器,服務器不對這個消息進行轉發,而只是判斷是否跨越了大格子,同時記錄下這個座標,做爲角色的當前位置。這一小段距離能夠取100左右,值取得越大時,消息發送頻率越小,但服務器的同步大格子跨越判斷和角色當前位置就越不精確。


  由於是直接同步的運行狀態,因此客戶端發給服務器的座標單位是3D座標系裏單位,而不是3D座標系格子的座標單位。這樣就更加精確,一點點距離的移動,都能準確同步。


f) 同步運動狀態的一個問題是,若是玩家操做很頻繁,好比快死了逃跑時,瘋狂地點地圖,這時運動狀態變化的很是快,若是每一個狀態的變化都同步給服務器,再加上廣播,那消息量是很大的。


  因此須要設置一個狀態同步的最短期,當運動狀態變化很快時,則將狀態變化的消息緩存在客戶端,同時加一個Timer跟蹤。當立刻有新的狀態變化消息出來時,則進行替換,同時更新Timer。當沒有狀態變化的消息出來時,Timer到時間了就會觸發,將緩存的狀態變化的消息,發給服務器。


  這樣經過消息緩存加上Timer的處理,既實現了運行狀態同步的最短期限制,也保證了最後有效的運行狀態會稍晚一點點發送給服務器。


4. 怪物的同步

  怪物的同步在傳統的端遊裏,是徹底由服務器的怪物AI系統觸發,客戶端只是純粹的接受服務器下發的怪物狀態數據。對於手機遊戲裏,因爲手機上很難出現像PC裏那樣的外掛,因此怪物的AI能夠考慮放在客戶端觸發,同時減小怪物的狀態同步。詳細說明以下:


a) 怪物的隨機移動不一樣步


  在地圖上,怪物都會有一個固定的位置。怪物沒有進入戰鬥狀態時,就會在這個固定位置的周圍走來走去,隨機的移動。這個隨機的移動由每一個客戶端本身控制,這樣怪物的隨機移動,就不用消息廣播進行同步了。


  因爲客戶端本身控制怪物的隨機走動,因此會出現不一樣客戶端裏,怪物位置不同的問題。但因爲怪物隨機移動的範圍較小,因此這個問題不是很明顯,在手機上是能夠接受的。角色打怪時,是扇形的傷害範圍,因此即便怪物座標在不一樣的客戶端有點不一致,打怪的效果也是能夠接受的。


b) 怪物的行爲同步


  當有角色攻擊被動怪物,或者進入主動怪物的視野範圍內時,怪物的AI就被這個角色所在的客戶端鎖定了,同時怪物進入攻擊狀態。攻擊的判斷徹底由鎖定怪物AI的客戶端進行處理,同時這個客戶端會將這個怪物的行爲上發到服務器,由服務器廣播給周圍的其餘玩家。


  怪物的AI鎖定,使用搶佔式,即誰最早發消息給服務器申請怪物的AI鎖定,誰就得到了怪物的控制權,直到怪物死亡或脫離戰鬥狀態。


  怪物能夠每進行一次攻擊,客戶端就發一個消息給服務器。這樣作,消息仍是有點多,特別是一羣怪圍着幾個角色進行攻擊時,消息廣播仍是有點多。因此能夠將狀態的概念向上擴大,只同步怪物在攻擊哪一個玩家,而不一樣步每一次的攻擊,而後由每一個客戶端根據怪物固定的攻擊速度各自去表現。這樣一個怪去攻擊一個玩家,就會只有一次消息廣播了。


c) 精英怪和BOSS怪的AI


  精英怪和BOSS怪因爲數量較少,並且比較重要,因此不能由客戶端來申請AI控制權,而是服務器根據某種策略來控制。所使用的策略能夠考慮角色的傷害值、防護值、角色與BOSS的距離遠近等,根據這些因素,服務器計算出BOSS怪當前最適合攻擊的對象(好比血量最少的玩家,最脆弱的法師等),而後將AI控制權發給那個客戶端,由那個客戶端控制攻擊行爲,同時經過消息讓服務器同步給其餘玩家。


  總結:怪物的同步方式的選擇,就是要儘可能減小消息的廣播,同時讓遊戲效果在可接受的範圍內。怪物AI的這個處理方式,其實是同時省去了遊戲服務器的怪物AI模塊(端遊通常是專門用的一個進程或者另一臺物理服務器來進行怪物AI的計算),從而簡化了MMO遊戲的開發難度,同時保證了較好的遊戲體驗。

相關文章
相關標籤/搜索