/branches/develop/Ivony.Core/Fluent/EnumerableExtension.cs

# · C# · 601 lines · 260 code · 178 blank · 163 comment · 90 complexity · ad29a9cdc68e27387e275c14c77dfa17 MD5 · raw file

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Collections;
  6. using System.Collections.ObjectModel;
  7. namespace Ivony.Fluent
  8. {
  9. /// <summary>
  10. /// 提供 IEnumerable 对象的扩展方法
  11. /// </summary>
  12. public static class EnumerableExtension
  13. {
  14. private static readonly Random random = new Random( DateTime.Now.Millisecond );
  15. /// <summary>
  16. /// 随机获取一个元素
  17. /// </summary>
  18. /// <typeparam name="T">元素类型</typeparam>
  19. /// <param name="source">元素集</param>
  20. /// <returns>随机挑选的一个元素</returns>
  21. public static T RandomElement<T>( this IEnumerable<T> source )
  22. {
  23. if ( source == null )
  24. throw new ArgumentNullException( "source" );
  25. if ( !source.Any() )
  26. throw new ArgumentException( "source", "序列不能为空" );
  27. return source.ElementAt( random.Next( source.Count() ) );
  28. }
  29. /// <summary>
  30. /// 对序列每一项执行指定操作
  31. /// </summary>
  32. /// <typeparam name="T">序列元素类型</typeparam>
  33. /// <param name="source">要执行操作的序列</param>
  34. /// <param name="action">要执行的操作</param>
  35. /// <returns>源序列</returns>
  36. public static IEnumerable<T> ForAll<T>( this IEnumerable<T> source, Action<T> action )
  37. {
  38. if ( source == null )
  39. throw new ArgumentNullException( "source" );
  40. if ( action == null )
  41. throw new ArgumentNullException( "action" );
  42. foreach ( T item in source )
  43. {
  44. action( item );
  45. }
  46. return source;
  47. }
  48. /// <summary>
  49. /// 将数组复制一份
  50. /// </summary>
  51. /// <typeparam name="T">数组元素类型</typeparam>
  52. /// <param name="array">源数组</param>
  53. /// <returns>复制的数组</returns>
  54. public static T[] Copy<T>( this T[] array )
  55. {
  56. T[] result = new T[array.Length];
  57. array.CopyTo( result, 0 );
  58. return result;
  59. }
  60. /// <summary>
  61. /// 对序列每一项执行指定操作
  62. /// </summary>
  63. /// <typeparam name="T">序列元素类型</typeparam>
  64. /// <param name="source">要执行操作的序列</param>
  65. /// <param name="action">要执行的操作</param>
  66. /// <returns>源序列</returns>
  67. public static IEnumerable<T> ForAll<T>( this IEnumerable<T> source, Action<T, int> action )
  68. {
  69. if ( source == null )
  70. throw new ArgumentNullException( "source" );
  71. if ( action == null )
  72. throw new ArgumentNullException( "action" );
  73. int i = 0;
  74. foreach ( T item in source )
  75. {
  76. action( item, i++ );
  77. }
  78. return source;
  79. }
  80. /// <summary>
  81. /// 对序列唯一项执行指定操作,若序列包含不止一个项,则抛出异常
  82. /// </summary>
  83. /// <typeparam name="T">序列元素类型</typeparam>
  84. /// <param name="source">要执行操作的序列</param>
  85. /// <param name="action">要执行的操作</param>
  86. /// <returns>源序列</returns>
  87. public static IEnumerable<T> ForSingle<T>( this IEnumerable<T> source, Action<T> action )
  88. {
  89. if ( source == null )
  90. throw new ArgumentNullException( "source" );
  91. if ( action == null )
  92. throw new ArgumentNullException( "action" );
  93. T singleItem = default( T );
  94. bool assigned = false;
  95. foreach ( var item in source )
  96. {
  97. if ( assigned )
  98. throw new InvalidOperationException( "序列包含一个以上的元素" );
  99. singleItem = item;
  100. assigned = true;
  101. }
  102. if ( assigned )
  103. action( singleItem );
  104. return source;
  105. }
  106. /// <summary>
  107. /// 对序列首项执行指定操作
  108. /// </summary>
  109. /// <typeparam name="T">序列元素类型</typeparam>
  110. /// <param name="source">要执行操作的序列</param>
  111. /// <param name="action">要执行的操作</param>
  112. /// <returns>源序列</returns>
  113. public static IEnumerable<T> ForFirst<T>( this IEnumerable<T> source, Action<T> action )
  114. {
  115. if ( source == null )
  116. throw new ArgumentNullException( "source" );
  117. if ( action == null )
  118. throw new ArgumentNullException( "action" );
  119. foreach ( var item in source )
  120. {
  121. action( item );
  122. break;
  123. }
  124. return source;
  125. }
  126. /// <summary>
  127. /// 对序列末项执行指定操作
  128. /// </summary>
  129. /// <typeparam name="T">序列元素类型</typeparam>
  130. /// <param name="source">要执行操作的序列</param>
  131. /// <param name="action">要执行的操作</param>
  132. /// <returns>源序列</returns>
  133. public static IEnumerable<T> ForLast<T>( this IEnumerable<T> source, Action<T> action )
  134. {
  135. if ( source == null )
  136. throw new ArgumentNullException( "source" );
  137. if ( action == null )
  138. throw new ArgumentNullException( "action" );
  139. T lastItem = default( T );
  140. bool assigned = false;
  141. foreach ( var item in source )
  142. {
  143. lastItem = item;
  144. assigned = true;
  145. }
  146. if ( assigned )
  147. action( lastItem );
  148. return source;
  149. }
  150. /// <summary>
  151. /// 从序列中去除为空(null)的项。
  152. /// </summary>
  153. /// <typeparam name="T"></typeparam>
  154. /// <param name="source"></param>
  155. /// <returns></returns>
  156. public static IEnumerable<T> NotNull<T>( this IEnumerable<T> source )
  157. {
  158. if ( source == null )
  159. throw new ArgumentNullException( "source" );
  160. return source.Where( item => item != null );
  161. }
  162. /// <summary>
  163. /// 确定序列是否确实有且只有一个元素
  164. /// </summary>
  165. /// <typeparam name="T">序列元素类型</typeparam>
  166. /// <param name="source">要检测的序列</param>
  167. /// <returns></returns>
  168. public static bool IsSingle<T>( this IEnumerable<T> source )
  169. {
  170. T obj;
  171. return IsSingle( source, out obj );
  172. }
  173. /// <summary>
  174. /// 确定序列是否确实有且只有一个元素
  175. /// </summary>
  176. /// <typeparam name="T">序列元素类型</typeparam>
  177. /// <param name="source">要检测的序列</param>
  178. /// <param name="element">如果有,则返回唯一的元素</param>
  179. /// <returns></returns>
  180. public static bool IsSingle<T>( this IEnumerable<T> source, out T element )
  181. {
  182. element = default( T );
  183. if ( source == null )
  184. throw new ArgumentNullException( "source" );
  185. var onlyone = false;
  186. foreach ( var item in source )
  187. {
  188. if ( onlyone )
  189. return false;
  190. onlyone = true;
  191. element = item;
  192. }
  193. return onlyone;
  194. }
  195. /// <summary>
  196. /// 指示指定的序列是否为 null 或者为空。
  197. /// </summary>
  198. /// <typeparam name="T">序列元素类型</typeparam>
  199. /// <param name="source">要检测的序列</param>
  200. /// <returns>是否为 null 或者为空</returns>
  201. public static bool IsNullOrEmpty<T>( this IEnumerable<T> source )
  202. {
  203. if ( source == null )
  204. return true;
  205. return !source.Any();
  206. }
  207. /// <summary>
  208. /// 将集合的每一项按顺序绑定到另一个集合
  209. /// </summary>
  210. /// <typeparam name="TSource">源集合元素类型</typeparam>
  211. /// <typeparam name="TTarget">目标集合元素类型</typeparam>
  212. /// <param name="source">源集合</param>
  213. /// <param name="targets">目标集合</param>
  214. /// <param name="binder">绑定方法</param>
  215. /// <returns>源集合</returns>
  216. public static IEnumerable<TSource> BindTo<TSource, TTarget>( this IEnumerable<TSource> source, IEnumerable<TTarget> targets, Action<TSource, TTarget> binder )
  217. {
  218. if ( source == null )
  219. throw new ArgumentNullException( "source" );
  220. if ( targets == null )
  221. throw new ArgumentNullException( "targets" );
  222. if ( binder == null )
  223. throw new ArgumentNullException( "binder" );
  224. return BindTo( source, targets, ( d, t, i ) => binder( d, t ) );
  225. }
  226. /// <summary>
  227. /// 将集合的每一项按顺序绑定到另一个集合
  228. /// </summary>
  229. /// <typeparam name="TSource">源集合元素类型</typeparam>
  230. /// <typeparam name="TTarget">目标集合元素类型</typeparam>
  231. /// <param name="source">源集合</param>
  232. /// <param name="targets">目标集合</param>
  233. /// <param name="binder">绑定方法</param>
  234. /// <returns>源集合</returns>
  235. public static IEnumerable<TSource> BindTo<TSource, TTarget>( this IEnumerable<TSource> source, IEnumerable<TTarget> targets, Action<TSource, TTarget, int> binder )
  236. {
  237. if ( source == null )
  238. throw new ArgumentNullException( "source" );
  239. if ( targets == null )
  240. throw new ArgumentNullException( "targets" );
  241. if ( binder == null )
  242. throw new ArgumentNullException( "binder" );
  243. BindCore( source, targets, binder );
  244. return source;
  245. }
  246. /// <summary>
  247. /// 将集合的每一项按顺序绑定到另一个集合
  248. /// </summary>
  249. /// <typeparam name="TSource">源集合元素类型</typeparam>
  250. /// <typeparam name="TTarget">目标集合元素类型</typeparam>
  251. /// <param name="source">源集合</param>
  252. /// <param name="targets">目标集合</param>
  253. /// <param name="defaultValue">当源集合元素不够时所采用的默认元素</param>
  254. /// <param name="binder">绑定方法</param>
  255. /// <returns>源集合</returns>
  256. public static IEnumerable<TSource> BindTo<TSource, TTarget>( this IEnumerable<TSource> source, IEnumerable<TTarget> targets, TSource defaultValue, Action<TSource, TTarget> binder )
  257. {
  258. if ( source == null )
  259. throw new ArgumentNullException( "source" );
  260. if ( targets == null )
  261. throw new ArgumentNullException( "targets" );
  262. if ( binder == null )
  263. throw new ArgumentNullException( "binder" );
  264. return BindTo( source, targets, defaultValue, ( d, t, i ) => binder( d, t ) );
  265. }
  266. /// <summary>
  267. /// 将集合的每一项按顺序绑定到另一个集合
  268. /// </summary>
  269. /// <typeparam name="TSource">源集合元素类型</typeparam>
  270. /// <typeparam name="TTarget">目标集合元素类型</typeparam>
  271. /// <param name="source">源集合</param>
  272. /// <param name="targets">目标集合</param>
  273. /// <param name="defaultValue">当源集合元素不够时所采用的默认元素</param>
  274. /// <param name="binder">绑定方法</param>
  275. /// <returns>源集合</returns>
  276. public static IEnumerable<TSource> BindTo<TSource, TTarget>( this IEnumerable<TSource> source, IEnumerable<TTarget> targets, TSource defaultValue, Action<TSource, TTarget, int> binder )
  277. {
  278. if ( source == null )
  279. throw new ArgumentNullException( "source" );
  280. if ( targets == null )
  281. throw new ArgumentNullException( "targets" );
  282. if ( binder == null )
  283. throw new ArgumentNullException( "binder" );
  284. BindCore( source, targets, defaultValue, binder );
  285. return source;
  286. }
  287. /// <summary>
  288. /// 将源集合的每一项按顺序绑定到集合
  289. /// </summary>
  290. /// <typeparam name="TSource">源集合元素类型</typeparam>
  291. /// <typeparam name="TTarget">目标集合元素类型</typeparam>
  292. /// <param name="source">源集合</param>
  293. /// <param name="targets">目标集合</param>
  294. /// <param name="binder">绑定方法</param>
  295. /// <returns>源集合</returns>
  296. public static IEnumerable<TTarget> BindFrom<TSource, TTarget>( this IEnumerable<TTarget> targets, IEnumerable<TSource> source, Action<TSource, TTarget> binder )
  297. {
  298. if ( source == null )
  299. throw new ArgumentNullException( "source" );
  300. if ( targets == null )
  301. throw new ArgumentNullException( "targets" );
  302. if ( binder == null )
  303. throw new ArgumentNullException( "binder" );
  304. return BindFrom( targets, source, ( s, t, i ) => binder( s, t ) );
  305. }
  306. /// <summary>
  307. /// 将源集合的每一项按顺序绑定到集合
  308. /// </summary>
  309. /// <typeparam name="TSource">源集合元素类型</typeparam>
  310. /// <typeparam name="TTarget">目标集合元素类型</typeparam>
  311. /// <param name="source">源集合</param>
  312. /// <param name="targets">目标集合</param>
  313. /// <param name="binder">绑定方法</param>
  314. /// <returns>源集合</returns>
  315. public static IEnumerable<TTarget> BindFrom<TSource, TTarget>( this IEnumerable<TTarget> targets, IEnumerable<TSource> source, Action<TSource, TTarget, int> binder )
  316. {
  317. if ( source == null )
  318. throw new ArgumentNullException( "source" );
  319. if ( targets == null )
  320. throw new ArgumentNullException( "targets" );
  321. if ( binder == null )
  322. throw new ArgumentNullException( "binder" );
  323. BindCore( source, targets, binder );
  324. return targets;
  325. }
  326. /// <summary>
  327. /// 将源集合的每一项按顺序绑定到集合
  328. /// </summary>
  329. /// <typeparam name="TSource">源集合元素类型</typeparam>
  330. /// <typeparam name="TTarget">目标集合元素类型</typeparam>
  331. /// <param name="source">源集合</param>
  332. /// <param name="targets">目标集合</param>
  333. /// <param name="defaultValue">当源集合元素不够时所采用的默认元素</param>
  334. /// <param name="binder">绑定方法</param>
  335. /// <returns>源集合</returns>
  336. public static IEnumerable<TTarget> BindFrom<TSource, TTarget>( this IEnumerable<TTarget> targets, IEnumerable<TSource> source, TSource defaultValue, Action<TSource, TTarget> binder )
  337. {
  338. if ( source == null )
  339. throw new ArgumentNullException( "source" );
  340. if ( targets == null )
  341. throw new ArgumentNullException( "targets" );
  342. if ( binder == null )
  343. throw new ArgumentNullException( "binder" );
  344. return BindFrom( targets, source, defaultValue, ( s, t, i ) => binder( s, t ) );
  345. }
  346. /// <summary>
  347. /// 将源集合的每一项按顺序绑定到集合
  348. /// </summary>
  349. /// <typeparam name="TSource">源集合元素类型</typeparam>
  350. /// <typeparam name="TTarget">目标集合元素类型</typeparam>
  351. /// <param name="source">源集合</param>
  352. /// <param name="targets">目标集合</param>
  353. /// <param name="defaultValue">当源集合元素不够时所采用的默认元素</param>
  354. /// <param name="binder">绑定方法</param>
  355. /// <returns>源集合</returns>
  356. public static IEnumerable<TTarget> BindFrom<TSource, TTarget>( this IEnumerable<TTarget> targets, IEnumerable<TSource> source, TSource defaultValue, Action<TSource, TTarget, int> binder )
  357. {
  358. if ( source == null )
  359. throw new ArgumentNullException( "source" );
  360. if ( targets == null )
  361. throw new ArgumentNullException( "targets" );
  362. if ( binder == null )
  363. throw new ArgumentNullException( "binder" );
  364. BindCore( source, targets, defaultValue, binder );
  365. return targets;
  366. }
  367. private static void BindCore<TSource, TTarget>( IEnumerable<TSource> source, IEnumerable<TTarget> targets, Action<TSource, TTarget, int> binder )
  368. {
  369. using ( var sourceIterator = source.GetEnumerator() )
  370. {
  371. using ( var targetIterator = targets.GetEnumerator() )
  372. {
  373. int index = 0;
  374. while ( sourceIterator.MoveNext() && targetIterator.MoveNext() )
  375. binder( sourceIterator.Current, targetIterator.Current, index++ );
  376. }
  377. }
  378. }
  379. private static void BindCore<TSource, TTarget>( IEnumerable<TSource> source, IEnumerable<TTarget> targets, TSource defaultValue, Action<TSource, TTarget, int> binder )
  380. {
  381. using ( var sourceIterator = source.GetEnumerator() )
  382. {
  383. using ( var targetIterator = targets.GetEnumerator() )
  384. {
  385. bool sourceEnded = false;
  386. while ( targetIterator.MoveNext() )
  387. {
  388. int index = 0;
  389. if ( !sourceEnded )
  390. sourceEnded = !sourceIterator.MoveNext();
  391. var dataItem = sourceEnded ? defaultValue : sourceIterator.Current;
  392. var targetItem = targetIterator.Current;
  393. binder( dataItem, targetItem, index++ );
  394. }
  395. }
  396. }
  397. }
  398. /// <summary>
  399. /// 创建只读枚举封装,避免集合类型强制类型转换后被修改
  400. /// </summary>
  401. /// <typeparam name="T">枚举元素类型</typeparam>
  402. /// <param name="enumerable">要创建只读枚举封装的集合</param>
  403. /// <returns>只读枚举封装</returns>
  404. public static IEnumerable<T> AsReadOnly<T>( this IEnumerable<T> enumerable )
  405. {
  406. if ( enumerable == null )
  407. return null;
  408. return new ReadOnlyEnumerable<T>( enumerable );
  409. }
  410. /// <summary>
  411. /// 创建只读列表封装,避免集合类型强制类型转换后被修改
  412. /// </summary>
  413. /// <typeparam name="T">列表元素类型</typeparam>
  414. /// <param name="list">要创建只读列表封装的集合</param>
  415. /// <returns>只读列表封装</returns>
  416. public static IList<T> AsReadOnly<T>( this IList<T> list )
  417. {
  418. if ( list == null )
  419. return null;
  420. return new ReadOnlyCollection<T>( list );
  421. }
  422. }
  423. }