Unity消息簡易框架 Advanced C# messenger

Unity消息簡易框架 Advanced C# messenger

【轉載 雨凇MOMO博客】 https://www.xuanyusong.com/archives/2165php

Tips

在尋找Unity下好用的消息事件框架的時候無心中發現該框架,十分簡易,但又用起來特別方便。公司VR遊戲中試了下,真滴方便。也不造輪子了,直接轉載雨凇大佬的博客,就當是Mark一下,方便之後學習。html

正文

Advanced CSharp Messenger 屬於C#事件的一種。 維基百科中由詳細的說明http://wiki.unity3d.com/index.php?title=Advanced_CSharp_Messenger 上週的一天剛巧有朋友問到我這一塊的知識,那麼我研究出來將它貼在博客中,幫助了他也幫助我本身!哇咔咔。框架

Advanced CSharp Messenger的特色能夠將遊戲對象作爲參數發送。到底Advanced CSharp Messenger有什麼用呢?先建立一個立方體對象,而後把Script腳本綁定在這個對象中。腳本中有一個方法叫DoSomething()。寫一段簡單的代碼,一般咱們在調用方法的時候須要這樣來寫。less

  1. private Script script; 
  2. void Awake() 

  3. GameObject cube = GameObject.Find("Cube"); 
  4. script = cube.GetComponent<Script>(); 

  5.  
  6. void Update() 

  7. if(Input.GetMouseButtonDown(0)) 

  8. script.DoSomething(); 


  9.  

代碼比較簡單,我就不註釋了。 原理就是先獲取遊戲對象,接着獲取腳本組件對象,最後經過腳本組件對象去調用對應腳本中的方法,這樣的調用方法咱們稱之爲直接調用。學習

這個例子中我只調用了一個對象的方法,若是說有成千上萬個對象,那麼這樣調用是否是感受本身的代碼很是的醜?由於你須要一個一個的獲取對象而後獲取腳本組件而後在調用方法。。。。。 (想一想都恐怖!!)this

下面咱們在用Advanced CSharp Messenger來實現事件的調用。按照維基百科中首先把Message.cs 和Callback.cs拷貝在你的工程中。spa

CallBack.cs3d

  1. public delegate void Callback()
  2. public delegate void Callback<T>(T arg1); 
  3. public delegate void Callback<T, U>(T arg1, U arg2); 
  4. public delegate void Callback<T, U, V>(T arg1, U arg2, V arg3); 

Message.cs代理

  1. /* 
  2. * Advanced C# messenger by Ilya Suzdalnitski. V1.0 
  3. *  
  4. * Based on Rod Hyde's "CSharpMessenger" and Magnus Wolffelt's "CSharpMessenger Extended". 
  5. *  
  6. * Features: 
  7. * Prevents a MissingReferenceException because of a reference to a destroyed message handler. 
  8. * Option to log all messages 
  9. * Extensive error detection, preventing silent bugs 
  10. *  
  11. * Usage examples: 
  12. 1. Messenger.AddListener<GameObject>("prop collected", PropCollected); 
  13. Messenger.Broadcast<GameObject>("prop collected", prop); 
  14. 2. Messenger.AddListener<float>("speed changed", SpeedChanged); 
  15. Messenger.Broadcast<float>("speed changed", 0.5f); 
  16. *  
  17. * Messenger cleans up its evenTable automatically upon loading of a new level. 
  18. *  
  19. * Don't forget that the messages that should survive the cleanup, should be marked with Messenger.MarkAsPermanent(string) 
  20. *  
  21. */ 
  22.  
  23. //#define LOG_ALL_MESSAGES 
  24. //#define LOG_ADD_LISTENER 
  25. //#define LOG_BROADCAST_MESSAGE 
  26. #define REQUIRE_LISTENER 
  27.  
  28. using System; 
  29. using System.Collections.Generic; 
  30. using UnityEngine; 
  31.  
  32. static internal class Messenger
  33. #region Internal variables 
  34.  
  35. //Disable the unused variable warning 
  36. #pragma warning disable 0414 
  37. //Ensures that the MessengerHelper will be created automatically upon start of the game. 
  38. static private MessengerHelper messengerHelper = ( new GameObject("MessengerHelper") ).AddComponent< MessengerHelper >(); 
  39. #pragma warning restore 0414 
  40.  
  41. static public Dictionary<string, Delegate> eventTable = new Dictionary<string, Delegate>(); 
  42.  
  43. //Message handlers that should never be removed, regardless of calling Cleanup 
  44. static public List< string > permanentMessages = new List< string > (); 
  45. #endregion 
  46. #region Helper methods 
  47. //Marks a certain message as permanent. 
  48. static public void MarkAsPermanent(string eventType)
  49. #if LOG_ALL_MESSAGES 
  50. Debug.Log("Messenger MarkAsPermanent \t\"" + eventType + "\""); 
  51. #endif 
  52.  
  53. permanentMessages.Add( eventType ); 

  54.  
  55. static public void Cleanup() 

  56. #if LOG_ALL_MESSAGES 
  57. Debug.Log("MESSENGER Cleanup. Make sure that none of necessary listeners are removed."); 
  58. #endif 
  59.  
  60. List< string > messagesToRemove = new List<string>(); 
  61.  
  62. foreach (KeyValuePair<string, Delegate> pair in eventTable) { 
  63. bool wasFound = false
  64.  
  65. foreach (string message in permanentMessages) { 
  66. if (pair.Key == message) { 
  67. wasFound = true
  68. break


  69.  
  70. if (!wasFound) 
  71. messagesToRemove.Add( pair.Key ); 

  72.  
  73. foreach (string message in messagesToRemove) { 
  74. eventTable.Remove( message ); 


  75.  
  76. static public void PrintEventTable() 

  77. Debug.Log("\t\t\t=== MESSENGER PrintEventTable ==="); 
  78.  
  79. foreach (KeyValuePair<string, Delegate> pair in eventTable) { 
  80. Debug.Log("\t\t\t" + pair.Key + "\t\t" + pair.Value); 

  81.  
  82. Debug.Log("\n"); 

  83. #endregion 
  84.  
  85. #region Message logging and exception throwing 
  86. static public void OnListenerAdding(string eventType, Delegate listenerBeingAdded)
  87. #if LOG_ALL_MESSAGES || LOG_ADD_LISTENER 
  88. Debug.Log("MESSENGER OnListenerAdding \t\"" + eventType + "\"\t{" + listenerBeingAdded.Target + " -> " + listenerBeingAdded.Method + "}"); 
  89. #endif 
  90.  
  91. if (!eventTable.ContainsKey(eventType)) { 
  92. eventTable.Add(eventType, null ); 

  93.  
  94. Delegate d = eventTable[eventType]; 
  95. if (d != null && d.GetType() != listenerBeingAdded.GetType()) { 
  96. throw new ListenerException(string.Format("Attempting to add listener with inconsistent signature for event type {0}. Current listeners have type {1} and listener being added has type {2}", eventType, d.GetType().Name, listenerBeingAdded.GetType().Name)); 


  97.  
  98. static public void OnListenerRemoving(string eventType, Delegate listenerBeingRemoved)
  99. #if LOG_ALL_MESSAGES 
  100. Debug.Log("MESSENGER OnListenerRemoving \t\"" + eventType + "\"\t{" + listenerBeingRemoved.Target + " -> " + listenerBeingRemoved.Method + "}"); 
  101. #endif 
  102.  
  103. if (eventTable.ContainsKey(eventType)) { 
  104. Delegate d = eventTable[eventType]; 
  105.  
  106. if (d == null) { 
  107. throw new ListenerException(string.Format("Attempting to remove listener with for event type \"{0}\" but current listener is null.", eventType)); 
  108. } else if (d.GetType() != listenerBeingRemoved.GetType()) { 
  109. throw new ListenerException(string.Format("Attempting to remove listener with inconsistent signature for event type {0}. Current listeners have type {1} and listener being removed has type {2}", eventType, d.GetType().Name, listenerBeingRemoved.GetType().Name)); 

  110. } else
  111. throw new ListenerException(string.Format("Attempting to remove listener for type \"{0}\" but Messenger doesn't know about this event type.", eventType)); 


  112.  
  113. static public void OnListenerRemoved(string eventType)
  114. if (eventTable[eventType] == null) { 
  115. eventTable.Remove(eventType); 


  116.  
  117. static public void OnBroadcasting(string eventType)
  118. #if REQUIRE_LISTENER 
  119. if (!eventTable.ContainsKey(eventType)) { 
  120. throw new BroadcastException(string.Format("Broadcasting message \"{0}\" but no listener found. Try marking the message with Messenger.MarkAsPermanent.", eventType)); 

  121. #endif 

  122.  
  123. static public BroadcastException CreateBroadcastSignatureException(string eventType)
  124. return new BroadcastException(string.Format("Broadcasting message \"{0}\" but listeners have a different signature than the broadcaster.", eventType)); 

  125.  
  126. public class BroadcastException : Exception
  127. public BroadcastException(string msg) 
  128. : base(msg)


  129.  
  130. public class ListenerException : Exception
  131. public ListenerException(string msg) 
  132. : base(msg)


  133. #endregion 
  134.  
  135. #region AddListener 
  136. //No parameters 
  137. static public void AddListener(string eventType, Callback handler)
  138. OnListenerAdding(eventType, handler); 
  139. eventTable[eventType] = (Callback)eventTable[eventType] + handler; 

  140.  
  141. //Single parameter 
  142. static public void AddListener<T>(string eventType, Callback<T> handler) { 
  143. OnListenerAdding(eventType, handler); 
  144. eventTable[eventType] = (Callback<T>)eventTable[eventType] + handler; 

  145.  
  146. //Two parameters 
  147. static public void AddListener<T, U>(string eventType, Callback<T, U> handler) { 
  148. OnListenerAdding(eventType, handler); 
  149. eventTable[eventType] = (Callback<T, U>)eventTable[eventType] + handler; 

  150.  
  151. //Three parameters 
  152. static public void AddListener<T, U, V>(string eventType, Callback<T, U, V> handler) { 
  153. OnListenerAdding(eventType, handler); 
  154. eventTable[eventType] = (Callback<T, U, V>)eventTable[eventType] + handler; 

  155. #endregion 
  156.  
  157. #region RemoveListener 
  158. //No parameters 
  159. static public void RemoveListener(string eventType, Callback handler)
  160. OnListenerRemoving(eventType, handler);  
  161. eventTable[eventType] = (Callback)eventTable[eventType] - handler; 
  162. OnListenerRemoved(eventType); 

  163.  
  164. //Single parameter 
  165. static public void RemoveListener<T>(string eventType, Callback<T> handler) { 
  166. OnListenerRemoving(eventType, handler); 
  167. eventTable[eventType] = (Callback<T>)eventTable[eventType] - handler; 
  168. OnListenerRemoved(eventType); 

  169.  
  170. //Two parameters 
  171. static public void RemoveListener<T, U>(string eventType, Callback<T, U> handler) { 
  172. OnListenerRemoving(eventType, handler); 
  173. eventTable[eventType] = (Callback<T, U>)eventTable[eventType] - handler; 
  174. OnListenerRemoved(eventType); 

  175.  
  176. //Three parameters 
  177. static public void RemoveListener<T, U, V>(string eventType, Callback<T, U, V> handler) { 
  178. OnListenerRemoving(eventType, handler); 
  179. eventTable[eventType] = (Callback<T, U, V>)eventTable[eventType] - handler; 
  180. OnListenerRemoved(eventType); 

  181. #endregion 
  182.  
  183. #region Broadcast 
  184. //No parameters 
  185. static public void Broadcast(string eventType)
  186. #if LOG_ALL_MESSAGES || LOG_BROADCAST_MESSAGE 
  187. Debug.Log("MESSENGER\t" + System.DateTime.Now.ToString("hh:mm:ss.fff") + "\t\t\tInvoking \t\"" + eventType + "\""); 
  188. #endif 
  189. OnBroadcasting(eventType); 
  190.  
  191. Delegate d; 
  192. if (eventTable.TryGetValue(eventType, out d)) { 
  193. Callback callback = d as Callback; 
  194.  
  195. if (callback != null) { 
  196. callback(); 
  197. } else
  198. throw CreateBroadcastSignatureException(eventType); 



  199.  
  200. //Single parameter 
  201. static public void Broadcast<T>(string eventType, T arg1) { 
  202. #if LOG_ALL_MESSAGES || LOG_BROADCAST_MESSAGE 
  203. Debug.Log("MESSENGER\t" + System.DateTime.Now.ToString("hh:mm:ss.fff") + "\t\t\tInvoking \t\"" + eventType + "\""); 
  204. #endif 
  205. OnBroadcasting(eventType); 
  206.  
  207. Delegate d; 
  208. if (eventTable.TryGetValue(eventType, out d)) { 
  209. Callback<T> callback = d as Callback<T>; 
  210.  
  211. if (callback != null) { 
  212. callback(arg1); 
  213. } else
  214. throw CreateBroadcastSignatureException(eventType); 



  215.  
  216. //Two parameters 
  217. static public void Broadcast<T, U>(string eventType, T arg1, U arg2) { 
  218. #if LOG_ALL_MESSAGES || LOG_BROADCAST_MESSAGE 
  219. Debug.Log("MESSENGER\t" + System.DateTime.Now.ToString("hh:mm:ss.fff") + "\t\t\tInvoking \t\"" + eventType + "\""); 
  220. #endif 
  221. OnBroadcasting(eventType); 
  222.  
  223. Delegate d; 
  224. if (eventTable.TryGetValue(eventType, out d)) { 
  225. Callback<T, U> callback = d as Callback<T, U>; 
  226.  
  227. if (callback != null) { 
  228. callback(arg1, arg2); 
  229. } else
  230. throw CreateBroadcastSignatureException(eventType); 



  231.  
  232. //Three parameters 
  233. static public void Broadcast<T, U, V>(string eventType, T arg1, U arg2, V arg3) { 
  234. #if LOG_ALL_MESSAGES || LOG_BROADCAST_MESSAGE 
  235. Debug.Log("MESSENGER\t" + System.DateTime.Now.ToString("hh:mm:ss.fff") + "\t\t\tInvoking \t\"" + eventType + "\""); 
  236. #endif 
  237. OnBroadcasting(eventType); 
  238.  
  239. Delegate d; 
  240. if (eventTable.TryGetValue(eventType, out d)) { 
  241. Callback<T, U, V> callback = d as Callback<T, U, V>; 
  242.  
  243. if (callback != null) { 
  244. callback(arg1, arg2, arg3); 
  245. } else
  246. throw CreateBroadcastSignatureException(eventType); 



  247. #endregion 

  248.  
  249. //This manager will ensure that the messenger's eventTable will be cleaned up upon loading of a new level. 
  250. public sealed class MessengerHelper : MonoBehaviour
  251. void Awake () 

  252. DontDestroyOnLoad(gameObject);  

  253.  
  254. //Clean up eventTable every time a new level loads. 
  255. public void OnDisable()
  256. Messenger.Cleanup(); 


而後就能夠開始使用了,Messager.Broadcast()這樣就比如咱們發送了一條廣播。rest

  1. void Update() 

  2. if(Input.GetMouseButtonDown(0)) 

  3. Messenger.Broadcast("Send"); 


在須要這條廣播的類中來接受它,一樣是剛剛說的Script類。接受廣播的標誌是 Messager.AddListener()參數1表示廣播的名稱,參數2表示廣播所調用的方法。

  1. using UnityEngine; 
  2. using System.Collections; 
  3.  
  4. public class Script : MonoBehaviour
  5.  
  6. void Awake() 

  7. Messenger.AddListener( "Send", DoSomething ); 

  8. public void DoSomething() 

  9. Debug.Log("DoSomething"); 


這樣一來,只要發送名稱爲」Send」的方法,就能夠在別的類中接收它了。

咱們在說說如何經過廣播來傳遞參數,這也是那天那個哥們主要問個人問題。(實際上是維基百科上寫的不是特別特別的清楚,那哥們誤解了)在Callback中能夠看出參數最多能夠是三個,參數的類型是任意類型,也就是說咱們不只能傳遞 int float bool 還能傳遞gameObject類型。

以下所示,發送廣播的時候傳遞了兩個參數,參數1是一個遊戲對象,參數2是一個int數值。

  1. void Update() 

  2. if(Input.GetMouseButtonDown(0)) 

  3. GameObject cube = GameObject.Find("Cube"); 
  4. Messenger.Broadcast<GameObject,int>("Send",cube,1980); 


而後是接受的地方 參數用<>存在一塊兒。遊戲對象也能夠完美的傳遞。

  1. using UnityEngine; 
  2. using System.Collections; 
  3.  
  4. public class Script : MonoBehaviour
  5.  
  6. void Awake() 

  7. Messenger.AddListener<GameObject,int>( "Send", DoSomething ); 

  8. public void DoSomething(GameObject obj,int i) 

  9. Debug.Log("name " + obj.name + " id =" + i); 


若是傳遞一個參數

兩個參數<T,T>

三個參數<T,T,T>

怎麼樣使用起來仍是挺簡單的吧?

我以爲項目中最好不要大量的使用代理事件這類的方法(根據需求而定),雖然可讓你的代碼很是的簡潔,可是它的效率不高大概比直接調用慢5-倍左右吧,就比如美好的東西必定都有瑕疵同樣。 還記得Unity自身也提供了一種發送消息的方法嗎?,用過的都知道效率也很是低下,雖然咱們看不到它具體實現的源碼是如何實現的,可是我以爲原理可能也是這樣的。 歡迎和你們一塊兒討論與學習。

相關文章
相關標籤/搜索