/WPFImage/BrainTechLLC.BaseLibrary/ThreadSafeObjects/ThreadSafeStack.cs

# · C# · 357 lines · 241 code · 52 blank · 64 comment · 23 complexity · 46606e19b9f4afb8c6f2f19a2a6244b5 MD5 · raw file

  1. // Original author contact info: Owen Emlen (owene_1998@yahoo.com)
  2. // Note: other individuals may also have contributed to this code
  3. // Project hosted on CodePlex.com as of 1/10/2009 at http://www.codeplex.com/EmlenMud
  4. using System.Runtime.Serialization;
  5. using System;
  6. using System.Collections.Generic;
  7. using System.Collections;
  8. using System.Collections.ObjectModel;
  9. using System.Collections.Specialized;
  10. using System.ComponentModel;
  11. #if USE_HYPER
  12. using Hyper.ComponentModel;
  13. #endif
  14. namespace BrainTechLLC.ThreadSafeObjects
  15. {
  16. /// <summary>
  17. /// A simple Stack that avoids using lock(). Fast/efficient to push/pop from multiple threads
  18. /// </summary>
  19. #if NO_SILVERLIGHT
  20. [TypeDescriptionProvider(typeof(HyperTypeDescriptionProvider))]
  21. #endif
  22. [Serializable]
  23. [DataContract]
  24. public class ThreadSafeStack<TItem> : Lockable, ICollection, INotifyCollectionChanged,
  25. IMultipleItems<TItem>, ISupportsCount where TItem : class
  26. {
  27. public override string ToString()
  28. {
  29. return _count.ToString();
  30. }
  31. [DataMember]
  32. public Stack<TItem> _stack = new Stack<TItem>();
  33. [DataMember]
  34. public int _count;
  35. public List<TItem> AllItems { get { return new List<TItem>(ArrayOfItems); } }
  36. public int Count { get { return _count; } }
  37. /// <summary>
  38. /// Pushes an item and returns the new count of items on the stack
  39. /// </summary>
  40. /// <param name="item"></param>
  41. /// <returns></returns>
  42. public int Push(TItem item)
  43. {
  44. // We need to be sure that no other threads simultaneously modify the shared _Stack
  45. // object during our Push operation
  46. AquireLock();
  47. {
  48. _stack.Push(item);
  49. _count++;
  50. }
  51. ReleaseLock();
  52. OnNotifyCollectionChanged(NotifyCollectionChangedAction.Add, item);
  53. return _count;
  54. }
  55. /// <summary>
  56. /// Pushes if a lock can be aquired, otherwise does nothing
  57. /// </summary>
  58. /// <param name="item"></param>
  59. /// <returns></returns>
  60. public int PushIfPossible(TItem item)
  61. {
  62. // We need to be sure that no other threads simultaneously modify the shared _queue
  63. // object during our push operation
  64. if (TryAquireLock())
  65. {
  66. _stack.Push(item);
  67. _count++;
  68. ReleaseLock();
  69. OnNotifyCollectionChanged(NotifyCollectionChangedAction.Add, item);
  70. }
  71. return _count;
  72. }
  73. /// <summary>
  74. /// Pushes multiple items and returns the new count of items on the stack
  75. /// </summary>
  76. /// <param name="items"></param>
  77. public int PushMultiple(List<TItem> items)
  78. {
  79. AquireLock();
  80. {
  81. foreach (TItem item in items)
  82. _stack.Push(item);
  83. _count += items.Count;
  84. }
  85. ReleaseLock();
  86. OnNotifyCollectionChanged(NotifyCollectionChangedAction.Add, items);
  87. return _count;
  88. }
  89. /// <summary>
  90. /// Pops multiple items (up to maxItems), places them in list 'items'
  91. /// Returns the number of items actually popped
  92. /// </summary>
  93. /// <param name="items"></param>
  94. /// <param name="maxItems"></param>
  95. /// <returns></returns>
  96. public int PopMultiple(List<TItem> items, int maxItems)
  97. {
  98. int popped = 0;
  99. AquireLock();
  100. {
  101. while (_count > 0 && popped < maxItems)
  102. {
  103. TItem item = _stack.Pop();
  104. items.Add(item);
  105. _count--;
  106. popped++;
  107. }
  108. }
  109. ReleaseLock();
  110. if (CollectionChanged != null)
  111. OnNotifyCollectionChanged(NotifyCollectionChangedAction.Remove, items);
  112. return popped;
  113. }
  114. /// <summary>
  115. /// Pops multiple items (up to maxItems), places them in array 'items'
  116. /// Returns the number of items actually popped
  117. /// </summary>
  118. /// <param name="items"></param>
  119. /// <param name="maxItems"></param>
  120. /// <returns></returns>
  121. public int PopMultiple(TItem[] items, int maxItems)
  122. {
  123. int popped = 0;
  124. AquireLock();
  125. {
  126. while (_count > 0 && popped < maxItems)
  127. {
  128. TItem item = _stack.Pop();
  129. items[popped] = item;
  130. popped++;
  131. _count--;
  132. }
  133. }
  134. ReleaseLock();
  135. if (CollectionChanged != null)
  136. OnNotifyCollectionChanged(NotifyCollectionChangedAction.Remove, new List<TItem>(items));
  137. return popped;
  138. }
  139. /// <summary>
  140. /// Pops an item from the stack. Returns null if stack was empty
  141. /// </summary>
  142. /// <returns></returns>
  143. public TItem PopItem()
  144. {
  145. TItem found;
  146. if (Pop(out found))
  147. {
  148. OnNotifyCollectionChanged(NotifyCollectionChangedAction.Remove, found);
  149. return found;
  150. }
  151. return null;
  152. }
  153. /// <summary>
  154. /// Pops an item from the stack. Returns false if stack was empty, returns true if item was successfully popped
  155. /// </summary>
  156. /// <param name="item"></param>
  157. /// <returns></returns>
  158. public bool Pop(out TItem item)
  159. {
  160. item = null;
  161. // We need to be sure that no other threads simultaneously modify the shared _Stack
  162. // object during our popped operation
  163. AquireLock();
  164. {
  165. if (_count > 0)
  166. {
  167. item = _stack.Pop();
  168. _count--;
  169. }
  170. }
  171. ReleaseLock();
  172. if (item != null)
  173. OnNotifyCollectionChanged(NotifyCollectionChangedAction.Remove, item);
  174. return (item != null);
  175. }
  176. /// <summary>
  177. /// Peeks at the top item on the stack. Returns null if stack is empty
  178. /// </summary>
  179. /// <returns></returns>
  180. public TItem PeekTop()
  181. {
  182. TItem item;
  183. if (_count > 0)
  184. {
  185. AquireLock();
  186. {
  187. if (_count > 0)
  188. {
  189. item = _stack.Peek();
  190. }
  191. else
  192. {
  193. item = null;
  194. }
  195. }
  196. ReleaseLock();
  197. }
  198. else
  199. {
  200. item = null;
  201. }
  202. return item;
  203. }
  204. /// <summary>
  205. /// Determines if an item is in the stack
  206. /// </summary>
  207. /// <param name="item"></param>
  208. /// <returns></returns>
  209. public bool IsInList(TItem item)
  210. {
  211. bool found;
  212. AquireLock();
  213. {
  214. found = _stack.Contains(item);
  215. }
  216. ReleaseLock();
  217. return found;
  218. }
  219. /// <summary>
  220. /// Returns an array of all items in the stack
  221. /// </summary>
  222. public TItem[] ArrayOfItems
  223. {
  224. get
  225. {
  226. TItem[] list;
  227. AquireLock();
  228. {
  229. list = _stack.ToArray();
  230. }
  231. ReleaseLock();
  232. return list;
  233. }
  234. }
  235. /// <summary>
  236. /// Clears the stack
  237. /// </summary>
  238. public void Clear()
  239. {
  240. IList<TItem> items = null;
  241. if (CollectionChanged != null)
  242. items = new List<TItem>(AllItems);
  243. AquireLock();
  244. {
  245. _stack.Clear();
  246. _count = 0;
  247. }
  248. ReleaseLock();
  249. OnNotifyCollectionChanged(NotifyCollectionChangedAction.Reset, items);
  250. }
  251. public void OnNotifyCollectionChanged(NotifyCollectionChangedAction action, IList<TItem> changedItems)
  252. {
  253. #if SILVERLIGHT
  254. #else
  255. if (CollectionChanged != null)
  256. {
  257. switch (action)
  258. {
  259. case NotifyCollectionChangedAction.Add:
  260. CollectionChanged(this, new NotifyCollectionChangedEventArgsEx<TItem>(action, changedItems));
  261. break;
  262. case NotifyCollectionChangedAction.Remove:
  263. CollectionChanged(this, new NotifyCollectionChangedEventArgsEx<TItem>(action, changedItems));
  264. break;
  265. case NotifyCollectionChangedAction.Replace:
  266. CollectionChanged(this, new NotifyCollectionChangedEventArgsEx<TItem>(action, changedItems[0], changedItems[1]));
  267. break;
  268. case NotifyCollectionChangedAction.Reset:
  269. CollectionChanged(this, new NotifyCollectionChangedEventArgsEx<TItem>(action));
  270. break;
  271. }
  272. }
  273. #endif
  274. }
  275. public void OnNotifyCollectionChanged(NotifyCollectionChangedAction action, TItem changedItem)
  276. {
  277. if (CollectionChanged != null)
  278. OnNotifyCollectionChanged(action, new List<TItem>() { changedItem });
  279. }
  280. #region INotifyCollectionChanged Members
  281. [field: NonSerialized]
  282. public event NotifyCollectionChangedEventHandler CollectionChanged;
  283. #endregion
  284. #region ICollection Members
  285. public void CopyTo(Array array, int index)
  286. {
  287. _stack.CopyTo((TItem[])array, index);
  288. }
  289. public bool IsSynchronized
  290. {
  291. get { return true; }
  292. }
  293. public object SyncRoot
  294. {
  295. get { return this; }
  296. }
  297. #endregion
  298. #region IEnumerable Members
  299. public IEnumerator GetEnumerator()
  300. {
  301. return _stack.GetEnumerator();
  302. }
  303. #endregion
  304. }
  305. }