Battle.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  1. using CommonAI.Zone;
  2. using CommonAI.Zone.ZoneEditor;
  3. using CommonLang;
  4. using CommonLang.Log;
  5. using System;
  6. using System.Collections.Generic;
  7. using System.Linq;
  8. using System.Text;
  9. namespace CommonTickBattle.Battle
  10. {
  11. public class Battle : InstanceZoneListener, IDisposable
  12. {
  13. protected readonly Logger log = LoggerFactory.GetLogger("ZoneNode");
  14. private readonly EditorTemplates mDataRoot;
  15. private readonly TemplateManager mTemplates;
  16. private readonly TBConfig mConfig;
  17. private readonly uint mUpdateInterval;
  18. private readonly uint mRequestLagging;
  19. private readonly uint mRequestLaggingTick;
  20. private readonly uint mResponseLagging;
  21. private readonly uint mResponseLaggingTick;
  22. //------------------------------------------------------------------------------------------------------------
  23. // 内部主线程命令 //
  24. private SceneData mSceneData;
  25. private EditorScene mZone;
  26. private SyncMessageQueue<Task> mTasks = new SyncMessageQueue<Task>();
  27. private ActionQueue mQueueActions = new ActionQueue();
  28. private bool mShutDown = false;
  29. private bool mStarted = false;
  30. private bool mRunning = false;
  31. //------------------------------------------------------------------------------------------------------------
  32. public Battle(EditorTemplates data_root, TBConfig cfg)
  33. {
  34. this.mDataRoot = data_root;
  35. this.mTemplates = data_root.Templates;
  36. this.mConfig = cfg;
  37. this.mUpdateInterval = cfg.GAME_FIXED_UPDATE_INTERVAL_MS;
  38. this.mRequestLagging = cfg.GAME_REQUEST_LAGGING_MS;
  39. this.mRequestLaggingTick = mRequestLagging / mUpdateInterval;
  40. this.mResponseLagging = cfg.GAME_RESPONSE_LAGGING_MS;
  41. this.mResponseLaggingTick = mResponseLagging / mUpdateInterval;
  42. }
  43. //------------------------------------------------------------------------------------------------------------
  44. public string Name { get { return (mSceneData != null) ? mSceneData.ToString() : "null"; } }
  45. public SceneData SceneData { get { return mSceneData; } }
  46. public EditorTemplates DataRoot { get { return mDataRoot; } }
  47. public TemplateManager Templates { get { return mTemplates; } }
  48. public int SceneID { get { return mSceneData.ID; } }
  49. public EditorScene Zone { get { return mZone; } }
  50. //------------------------------------------------------------------------------------------------------------
  51. /// <summary>
  52. /// 总共跑了多少帧
  53. /// </summary>
  54. public ulong CurrentTick { get { return mZone.Tick; } }
  55. /// <summary>
  56. /// 每帧多少毫秒
  57. /// </summary>
  58. public uint UpdateIntervalMS { get { return mUpdateInterval; } }
  59. /// <summary>
  60. /// Local->Host 超时时间(毫秒)
  61. /// </summary>
  62. public uint RequestLaggingMS { get { return mRequestLagging; } }
  63. /// <summary>
  64. /// Host->Local 超时时间(毫秒)
  65. /// </summary>
  66. public uint ResponseLaggingMS { get { return mResponseLagging; } }
  67. /// <summary>
  68. /// Local->Host 超时时间(帧)
  69. /// </summary>
  70. public uint RequestLaggingTick { get { return mRequestLaggingTick; } }
  71. /// <summary>
  72. /// Host->Local 超时时间(帧)
  73. /// </summary>
  74. public uint ResponseLaggingTick { get { return mResponseLaggingTick; } }
  75. //------------------------------------------------------------------------------------------------------------
  76. public override string ToString()
  77. {
  78. if (mSceneData != null)
  79. {
  80. return mSceneData.Name + "(" + mSceneData.ID + ")";
  81. }
  82. return "Unable to load scene";
  83. }
  84. //------------------------------------------------------------------------------------------------------------
  85. /// <summary>
  86. /// 房间初始化
  87. /// </summary>
  88. public void Start(SceneData data)
  89. {
  90. lock (this)
  91. {
  92. if (mStarted) { return; }
  93. this.mStarted = true;
  94. // 解析游戏服创建房间信息 //
  95. this.mSceneData = data;
  96. {
  97. // 构造战斗场景 //
  98. this.mZone = TemplateManager.Factory.CreateEditorScene(this.Templates, this, mSceneData);
  99. // 非全屏同步,每个Client负责维护自己需要的队列 //
  100. this.mZone.SyncPos = false;
  101. this.mZone.IsSyncZ = false;
  102. // 半同步,场景不能大于255 //
  103. this.mZone.IsHalfSync = true;
  104. }
  105. this.mRunning = true;
  106. // 创建游戏主循环Timer //
  107. this.QueueTask(() =>
  108. {
  109. OnStarted();
  110. if (event_OnZoneStart != null)
  111. {
  112. event_OnZoneStart.Invoke(this);
  113. }
  114. });
  115. }
  116. }
  117. /// <summary>
  118. /// 开始异步关闭房间
  119. /// </summary>
  120. public void Stop()
  121. {
  122. lock (this)
  123. {
  124. mShutDown = true;
  125. }
  126. }
  127. public void Dispose()
  128. {
  129. lock (this)
  130. {
  131. if (mRunning)
  132. {
  133. throw new Exception("Battle is running");
  134. }
  135. this.OnDispose();
  136. this.DisposeEvents();
  137. if (mZone != null)
  138. {
  139. this.mZone.Dispose();
  140. }
  141. }
  142. }
  143. //------------------------------------------------------------------------------------------------------------
  144. /// <summary>
  145. /// 战斗场景主逻辑更新//
  146. /// </summary>
  147. public virtual void Update()
  148. {
  149. lock (this)
  150. {
  151. if (mRunning)
  152. {
  153. int intervalMS = (int)mUpdateInterval;
  154. mTasks.ProcessMessages(do_task);
  155. ProcessAction();
  156. OnBeginUpdate();
  157. try
  158. {
  159. if (intervalMS > 0)
  160. {
  161. mZone.Update(intervalMS);
  162. this.OnZoneUpdate();
  163. }
  164. }
  165. catch (Exception err)
  166. {
  167. log.Error(err.Message, err);
  168. OnError(err);
  169. }
  170. OnEndUpdate();
  171. if (mShutDown)
  172. {
  173. mRunning = false;
  174. OnStopped();
  175. if (event_OnZoneStop != null)
  176. {
  177. event_OnZoneStop.Invoke(this);
  178. }
  179. }
  180. }
  181. }
  182. }
  183. void InstanceZoneListener.onEventHandler(Event e)
  184. {
  185. this.OnZoneEventHandler(e);
  186. }
  187. //------------------------------------------------------------------------------------------------------------
  188. #region FutureAction
  189. /// <summary>
  190. /// 想在未来某个时间点执行一个动作
  191. /// </summary>
  192. /// <param name="action"></param>
  193. /// <param name="tick">未来某个时间点</param>
  194. /// <returns></returns>
  195. protected virtual bool PushAction(CommonAI.Zone.Action action, ulong tick)
  196. {
  197. lock (this)
  198. {
  199. if (tick > CurrentTick)
  200. {
  201. mQueueActions.Push(new FutureAction(action, tick));
  202. return true;
  203. }
  204. else
  205. {
  206. log.ErrorFormat("指令已过期 : AT={0} : {1}", tick, action);
  207. }
  208. return false;
  209. }
  210. }
  211. private void ProcessAction()
  212. {
  213. while (mQueueActions.Count > 0)
  214. {
  215. var req = mQueueActions.Pop(Zone.Tick);
  216. if (req != null)
  217. {
  218. Zone.pushAction(req.action);
  219. }
  220. else
  221. {
  222. break;
  223. }
  224. }
  225. }
  226. private class FutureAction
  227. {
  228. public readonly CommonAI.Zone.Action action;
  229. public readonly ulong action_tick;
  230. public FutureAction(CommonAI.Zone.Action action, ulong tick)
  231. {
  232. this.action = action;
  233. this.action_tick = tick;
  234. }
  235. }
  236. private class ActionQueue : IComparer<FutureAction>
  237. {
  238. private List<FutureAction> queue = new List<FutureAction>();
  239. public int Count { get { return queue.Count; } }
  240. public void Push(FutureAction req)
  241. {
  242. queue.Add(req);
  243. queue.Sort(this);
  244. }
  245. public FutureAction Pop(ulong tick)
  246. {
  247. if (queue.Count > 0)
  248. {
  249. var req = queue[0];
  250. if (req.action_tick == tick)
  251. {
  252. queue.RemoveAt(0);
  253. return req;
  254. }
  255. }
  256. return null;
  257. }
  258. public int Compare(FutureAction x, FutureAction y)
  259. {
  260. return (int)x.action_tick - (int)y.action_tick;
  261. }
  262. }
  263. #endregion
  264. //---------------------------------------------------------------------------------------
  265. protected internal delegate void Task();
  266. /// <summary>
  267. /// 保证在Task内部执行的代码线程安全
  268. /// </summary>
  269. /// <param name="task"></param>
  270. protected internal void QueueTask(Task task)
  271. {
  272. mTasks.Enqueue(task);
  273. }
  274. private void do_task(Task task)
  275. {
  276. task.Invoke();
  277. }
  278. //---------------------------------------------------------------------------------------------------
  279. protected virtual void OnStarted() { }
  280. protected virtual void OnStopped() { }
  281. protected virtual void OnZoneUpdate() { }
  282. protected virtual void OnBeginUpdate() { }
  283. protected virtual void OnEndUpdate() { }
  284. protected virtual void OnZoneEventHandler(Event e) { }
  285. protected virtual void OnError(Exception err) { }
  286. protected virtual void OnDispose() { }
  287. //------------------------------------------------------------------------------------------------------------
  288. protected virtual void DisposeEvents()
  289. {
  290. this.event_OnZoneStart = null;
  291. this.event_OnZoneStop = null;
  292. }
  293. private Action<Battle> event_OnZoneStart;
  294. private Action<Battle> event_OnZoneStop;
  295. public event Action<Battle> OnBattleStart { add { event_OnZoneStart += value; } remove { event_OnZoneStart -= value; } }
  296. public event Action<Battle> OnBattleStop { add { event_OnZoneStop += value; } remove { event_OnZoneStop -= value; } }
  297. //---------------------------------------------------------------------------------------------------
  298. }
  299. }