LJCache.cs 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Diagnostics;
  4. using System.Threading;
  5. using LJLib.D;
  6. namespace LJLib
  7. {
  8. /// <summary>
  9. /// 缓存,超时自动清除,线程安全
  10. /// </summary>
  11. public sealed class LJCache<TKey, TValue> : Dictionary<TKey, TValue>
  12. {
  13. /// <summary>
  14. /// 自动延长分钟数
  15. /// </summary>
  16. public double DefaultAddMinutes
  17. {
  18. get { return _defaultAddMinutes;}
  19. set { _defaultAddMinutes = value; }
  20. }
  21. /// <summary>
  22. /// 添加键值对,
  23. /// </summary>
  24. /// <param name="key"></param>
  25. /// <param name="value"></param>
  26. /// <param name="minutes">超时分钟数</param>
  27. public void Add(TKey key, TValue value, double minutes)
  28. {
  29. var deadline = DateTime.Now.AddMinutes(minutes);
  30. Add(key, value, deadline);
  31. }
  32. /// <summary>
  33. /// 添加永久对象或者按默认时间添加
  34. /// </summary>
  35. /// <param name="key"></param>
  36. /// <param name="value"></param>
  37. public new void Add(TKey key, TValue value)
  38. {
  39. Add(key, value, _defaultAddMinutes);
  40. }
  41. public new TValue this[TKey key]
  42. {
  43. get
  44. {
  45. lock (syncRoot)
  46. {
  47. SetDeadLine(key, DateTime.Now.AddMinutes(DefaultAddMinutes));
  48. return base[key];
  49. }
  50. }
  51. set
  52. {
  53. lock (syncRoot)
  54. {
  55. SetDeadLine(key, DateTime.Now.AddMinutes(DefaultAddMinutes));
  56. base[key] = value;
  57. }
  58. }
  59. }
  60. public new bool ContainsKey(TKey key)
  61. {
  62. lock (syncRoot)
  63. {
  64. var rslt = base.ContainsKey(key);
  65. if (rslt)
  66. {
  67. SetDeadLine(key, DateTime.Now.AddMinutes(DefaultAddMinutes));
  68. }
  69. return rslt;
  70. }
  71. }
  72. #region 稀有成员实现逻辑
  73. /// <summary>
  74. /// 添加键值对,设置超时时间
  75. /// </summary>
  76. /// <param name="key"></param>
  77. /// <param name="value"></param>
  78. /// <param name="deadline">超时时间</param>
  79. private void Add(TKey key, TValue value, DateTime deadline)
  80. {
  81. lock (syncRoot)
  82. {
  83. base[key] = value;
  84. SetDeadLine(key, deadline);
  85. }
  86. }
  87. /// <summary>
  88. /// 手动设置超时时间
  89. /// </summary>
  90. /// <param name="key"></param>
  91. /// <param name="deadline"></param>
  92. /// <returns></returns>
  93. private void SetDeadLine(TKey key, DateTime deadline)
  94. {
  95. lock (syncRoot)
  96. {
  97. deadline = deadline.AddMilliseconds(-deadline.Millisecond);
  98. if (_deadlines == null)
  99. {
  100. _deadlines = new Dictionary<TKey, DateTime>();
  101. _ix_keys = new SortedList<DateTime, HashSet<TKey>>();
  102. }
  103. if (_deadlines.ContainsKey(key))
  104. {
  105. var olddt = _deadlines[key];
  106. _ix_keys[olddt].Remove(key);
  107. if (_ix_keys[olddt].Count == 0)
  108. {
  109. _ix_keys.Remove(olddt);
  110. }
  111. }
  112. _deadlines[key] = deadline;
  113. if (!_ix_keys.ContainsKey(deadline))
  114. {
  115. _ix_keys[deadline] = new HashSet<TKey>();
  116. }
  117. if (!_ix_keys[deadline].Contains(key))
  118. {
  119. _ix_keys[deadline].Add(key);
  120. }
  121. if (_timer == null)
  122. {
  123. _timer = new Timer((state) =>
  124. {
  125. try
  126. {
  127. //DebugHelper.GetTotalMemory("LJCache.Remove.开始");
  128. lock (syncRoot)
  129. {
  130. var dtNow = DateTime.Now;
  131. List<DateTime> dts = new List<DateTime>();
  132. for (int i = 0; _ix_keys != null && i < _ix_keys.Keys.Count && _ix_keys.Keys[i] <= dtNow; i++)
  133. {
  134. dts.Add(_ix_keys.Keys[i]);
  135. foreach (var oldkey in _ix_keys.Values[i])
  136. {
  137. _deadlines.Remove(oldkey);
  138. base.Remove(oldkey);
  139. }
  140. }
  141. foreach (var dt in dts)
  142. {
  143. _ix_keys.Remove(dt);
  144. }
  145. }
  146. //DebugHelper.GetTotalMemory("LJCache.Remove.结束");
  147. }
  148. catch (Exception ex)
  149. {
  150. Trace.Write(ex);
  151. }
  152. }, null, 1000, 1000);
  153. }
  154. }
  155. }
  156. /// <summary>
  157. /// 线路同步对象
  158. /// </summary>
  159. private object syncRoot = new object();
  160. /// <summary>
  161. /// 缓存清除定时器
  162. /// </summary>
  163. private Timer _timer;
  164. /// <summary>
  165. /// 各键值对的销毁时间
  166. /// </summary>
  167. private Dictionary<TKey, DateTime> _deadlines;
  168. /// <summary>
  169. /// 按时间顺序过期
  170. /// </summary>
  171. private SortedList<DateTime, HashSet<TKey>> _ix_keys;
  172. /// <summary>
  173. /// 自动延长分钟数
  174. /// </summary>
  175. private double _defaultAddMinutes = 5;
  176. public LJCache(IEqualityComparer<TKey> comparer) : base(comparer) { }
  177. public LJCache() : base() { }
  178. #endregion
  179. }
  180. }