ObjectPool.cs 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. using CommonLang.Concurrent;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.IO;
  5. using System.Linq;
  6. using System.Text;
  7. namespace CommonLang
  8. {
  9. public static class ObjectPoolStatus
  10. {
  11. internal static AtomicLong s_total_count = new AtomicLong(0);
  12. internal static AtomicLong s_total_inactive = new AtomicLong(0);
  13. public static long TotalCount
  14. {
  15. get { return s_total_count.Value; }
  16. }
  17. public static long TotalInactive
  18. {
  19. get { return s_total_inactive.Value; }
  20. }
  21. public static long TotalActive
  22. {
  23. get { return TotalCount - TotalInactive; }
  24. }
  25. }
  26. public class ObjectPool<T>
  27. {
  28. public delegate T ActionCreater();
  29. private readonly Stack<T> m_Stack = new Stack<T>();
  30. private readonly ActionCreater m_ActionOnCreate;
  31. private readonly Action<T> m_ActionOnGet;
  32. private readonly Action<T> m_ActionOnRelease;
  33. private AtomicLong count = new AtomicLong(0);
  34. public long countAll
  35. {
  36. get { return count.Value; }
  37. }
  38. public long countInactive
  39. {
  40. get { return this.m_Stack.Count; }
  41. }
  42. public long countActive
  43. {
  44. get { return this.countAll - this.countInactive; }
  45. }
  46. public ObjectPool(ActionCreater actionCreate = null, Action<T> actionOnGet = null, Action<T> actionOnRelease = null)
  47. {
  48. this.m_ActionOnGet = actionOnGet;
  49. this.m_ActionOnRelease = actionOnRelease;
  50. this.m_ActionOnCreate = actionCreate;
  51. }
  52. public T Get()
  53. {
  54. lock (this)
  55. {
  56. T t;
  57. if (this.m_Stack.Count == 0)
  58. {
  59. if (m_ActionOnCreate != null)
  60. {
  61. t = m_ActionOnCreate();
  62. }
  63. else
  64. {
  65. t = ((default(T) == null) ? Activator.CreateInstance<T>() : default(T));
  66. }
  67. this.count++;
  68. ObjectPoolStatus.s_total_count++;
  69. }
  70. else
  71. {
  72. t = this.m_Stack.Pop();
  73. ObjectPoolStatus.s_total_inactive--;
  74. }
  75. if (this.m_ActionOnGet != null)
  76. {
  77. this.m_ActionOnGet(t);
  78. }
  79. return t;
  80. }
  81. }
  82. public void Release(T element)
  83. {
  84. lock (this)
  85. {
  86. if (this.m_Stack.Count > 0 && object.ReferenceEquals(this.m_Stack.Peek(), element))
  87. {
  88. throw new Exception("Internal error. Trying to destroy object that is already released to pool.");
  89. }
  90. if (this.m_ActionOnRelease != null)
  91. {
  92. this.m_ActionOnRelease(element);
  93. }
  94. this.m_Stack.Push(element);
  95. ObjectPoolStatus.s_total_inactive++;
  96. }
  97. }
  98. }
  99. public static class ListObjectPool<T>
  100. {
  101. private static readonly ObjectPool<List<T>> s_ListPool = new ObjectPool<List<T>>(s_ListPool_OnCreate, null, s_ListPool_OnRelease);
  102. private static List<T> s_ListPool_OnCreate()
  103. {
  104. return new AutoReleaseList<T>();
  105. }
  106. private static void s_ListPool_OnRelease(List<T> l)
  107. {
  108. l.Clear();
  109. }
  110. public static AutoReleaseList<T> AllocAutoRelease()
  111. {
  112. AutoReleaseList<T> ret = s_ListPool.Get() as AutoReleaseList<T>;
  113. return ret;
  114. }
  115. public static AutoReleaseList<T> AllocAutoRelease(IEnumerable<T> added)
  116. {
  117. AutoReleaseList<T> ret = s_ListPool.Get() as AutoReleaseList<T>;
  118. if (added != null) { ret.AddRange(added); }
  119. return ret;
  120. }
  121. public static AutoReleaseList<T> AllocAutoRelease(T[] added)
  122. {
  123. AutoReleaseList<T> ret = s_ListPool.Get() as AutoReleaseList<T>;
  124. if (added != null) { ret.AddRange(added); }
  125. return ret;
  126. }
  127. public static AutoReleaseList<T> AllocAutoRelease(int capacity)
  128. {
  129. AutoReleaseList<T> ret = s_ListPool.Get() as AutoReleaseList<T>;
  130. if (capacity != 0)
  131. {
  132. ret.Capacity = capacity;
  133. }
  134. return ret;
  135. }
  136. public static void Release(List<T> toRelease)
  137. {
  138. s_ListPool.Release(toRelease);
  139. }
  140. public class AutoReleaseList<R> : List<R>, IDisposable
  141. {
  142. internal AutoReleaseList() { }
  143. void IDisposable.Dispose()
  144. {
  145. ListObjectPool<R>.Release(this);
  146. }
  147. }
  148. }
  149. public static class MemoryStreamObjectPool
  150. {
  151. private static readonly ObjectPool<AutoRelease> s_Pool = new ObjectPool<AutoRelease>(s_ListPool_OnCreate, null, null);
  152. private static AutoRelease s_ListPool_OnCreate()
  153. {
  154. return new AutoRelease();
  155. }
  156. public static AutoRelease AllocAutoRelease()
  157. {
  158. AutoRelease ret = s_Pool.Get() as AutoRelease;
  159. return ret;
  160. }
  161. public static AutoRelease AllocAutoRelease(int capacity)
  162. {
  163. AutoRelease ret = s_Pool.Get() as AutoRelease;
  164. ret.Capacity = capacity;
  165. return ret;
  166. }
  167. public static AutoRelease AllocAutoRelease(byte[] buffer)
  168. {
  169. AutoRelease ret = s_Pool.Get() as AutoRelease;
  170. IO.IOUtil.WriteToEnd(ret, buffer, 0, buffer.Length);
  171. ret.Position = 0;
  172. return ret;
  173. }
  174. public static AutoRelease AllocAutoRelease(byte[] buffer, int offset, int length)
  175. {
  176. AutoRelease ret = s_Pool.Get() as AutoRelease;
  177. IO.IOUtil.WriteToEnd(ret, buffer, offset, length);
  178. ret.Position = 0;
  179. return ret;
  180. }
  181. public static void Release(AutoRelease toRelease)
  182. {
  183. s_Pool.Release(toRelease);
  184. }
  185. public class AutoRelease : MemoryStream
  186. {
  187. internal AutoRelease() { }
  188. protected override void Dispose(bool disposing)
  189. {
  190. this.Position = 0;
  191. this.SetLength(0);
  192. MemoryStreamObjectPool.Release(this);
  193. }
  194. }
  195. }
  196. }